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内 容 简介 


本 书 放 析 了 50 个 典型 的 程序 员 面 试题 ， 从 基础 知识 、 代 码 质量 、 解 
题 思路 、 优 化 效率 和 综合 能 力 五 个 方面 系统 整理 了 影响 面试 的 5 个 要 
扩 。 全 书 分 为 7 划 ， 主 要 包括 面试 的 流程 ， 讨 论 面 试 流程 中 每 一 坏 节 需 
要 注意 的 问题 ， 面 试 需要 的 基础 知识 ， 从 编程 语言 、 数 据 结构 及 算法 三 
方面 忠 结 了 程序 员 面 试 的 知识 点 ;高 质量 的 代码 ， 讨 论 影 响 代 人 码 质量 的 
的 3 个 要 素 〈 规 范 性 、 完 整 性 和 重 棒 性 ) ， 强 调 高 质量 的 代码 除了 能 够 
完成 基本 的 功能 之 外 ， 还 能 考虑 到 特殊 情况 并 对 非法 输入 进行 合理 的 处 
理 ;， 解 决 面 试题 的 思路 ， 总 结 在 编程 面试 中 解决 难题 的 冲 用 思路 ， 如 宁 
在 面试 过 程 中 过 到 了 复杂 的 难题 ， 应 聘 者 可 以 利用 和 画图、 举例 和 分 解 复 
杂 问 题 3 种 方法 化 紧 为 位， 先 形 成 清晰 的 思路 再 动手 编程 ， 优 化 时 间 和 
空间 效率 ， 介 绍 如 何 优化 代码 的 时 间 效 率 和 空间 效率 ， 读 完 这 一 章 读 者 
将 学 会 常用 的 优化 时 间 效 率 及 空间 换 时 间 的 常用 算法 ， 从 而 在 面试 中 找 
到 最 优 的 解法 ;面试 中 的 各 种 能 力 ， 本 章 总 结 应 聘 者 在 面试 过 程 中 如 何 
表现 学 习 能 力 和 沟通 能 力 ， 并 通过 具体 的 面试 题 讨 论 如 何 培 养 知识 迁移 
能 力 、 抽 象 建 模 能 力 和 发 散 思 维 能 力 ;， 两 个 面试 案例 ， 这 两 个 案例 总 结 
了 应 聘 者 在 面试 过 程 中 哪些 举动 是 不 好 的 行为 ， 而 哪些 表现 又 是 面试 官 
所 期 竺 的 行为 。 

本 书 适合 即将 走向 工作 岗位 的 大 学 生 阅 读 ， 也 适合 作为 正在 应 聘 软 
件 行业 的 相关 就 业 人 员 和 计算 机 爱好 者 的 参考 书 。 

















未 经 许可 ， 不 得 以 任何 方式 复制 或 抄袭 本 书 的 任何 部 分 。 
版 权 所 有 ， 侵 权 必 完 。 


图 书 在 版 编目 CIP) 数据 

剑 指 Offer: 名 企 面试 官 精 讲 典 型 编程 题 / 何 海 涛 著 . 一 北京 : 电子 工业 
出 版 社 ，2012.1 

ISBN 978-7-121-14875-0 


I. ©]... I. OWH... M. 也 程序 设计 一 工程 技术 人 员 一 资格 考试 
一 习题 集 IV. TP311.1-44 





中 国 版 本 图 书馆 CIP 数 据 核 字 (2011) 第 215025 号 





策划 编辑 张 春 雨 
责任 编辑 ， 李 云 静 
特约 编辑 : 赵 树 刚 
EN Wh 北京 丰 源 印刷 厂 
装 订 : 三 河 市 鹏 成 印 业 有 限 公司 
出 版 发 行 ， 电子 工业 出 版 社 
北京 市 海淀 区 万 寿 路 173 人 信箱” 邮编 ”100036 
H AS: 787x980 1/16 印张: 17.25 字数 :354 千 字 
印 YR: 2012 年 1 月 第 1 次 印刷 
印 数 : 4000 册 定价 : 45.00 元 


凡 所 购买 电子 工业 出 版 社 图 书 有 缺损 问题 ， 请 癌 购 买书 店 调换 。 千 


书店 售 缺 ， 请 与 本 社 发 行 部 联系 ， 联 系 及 邮购 电话 : (010) 
88254888 。 

质量 投诉 请 发 邮件 至 zlts@phei.com.cn， 盗 版 侵权 举报 请 发 邮件 至 
dbqq@phei.com.cn. 


服务 热线 : (010) 88258888. 


推荐 序 一 


海 族 2008 年 在 我 的 团队 做 过 软件 开发 工程 师 。 他 是 一 个 很 细心 的 员 
工 ， 对 面试 这 个 话题 很 感 兴 趣 ， 经 常 和 我 及 其 他 员工 讨论 ， 积 累 了 很 多 
面试 方面 的 技巧 和 经 验 。 他 曾 跟 我 所 过 想 要 写本 有 关 面 试 的 书 ， 三 年 过 
后 他 把 书写 出 来 了 ! 他 是 一 个 有 目标 、 有 了 耐心 和 持久 力 的 人 。 

我 在 微软 做 了 很 多 年 的 面试 官 ， 后 面 七 年 多 作为 把 天 面试 官 也 面试 
了 很 多 应 聘 者 。 应 聘 者 要 想 做 好 面试 ， 确 实 应 把 面试 当 作 一 门 技巧 来 学 
习 ， 更 重要 的 是 要 提高 自身 的 能 力 。 我 遇 到 很 多 应 试 者 可 能 自身 能 力也 
不 差 但 因为 不 懂得 怎样 回答 提问 ， 不 能 很 好 发 挥 。 也 有 很 多 校园 来 的 应 
聘 者 也 学 过 数据 结构 和 算法 分 析 ， 可 是 到 处 理 具体 问题 时 不 能 用 学 过 的 
知识 来 有 效 地 解决 问题 。 这 些 朋友 读 读 海 涛 的 这 本 书 ， 会 很 受益 ， 在 面 
试 中 的 发 挥 也 会 有 很 大 提高 。 这 本 书 也 可 以 作为 很 好 的 教学 补充 资料 ， 
让 学 生 不 只 学 到 书本 知识 ， 也 学 到 解决 问题 的 能 力 。 

在 回 我 汇报 的 员工 中 有 面试 发 挥 很 好 但 工作 平平 的 ， 也 有 面试 一 般 
但 工作 优秀 的 。 对 于 退 求 职业 发 展 的 人 来 说 ， 通 过 面试 只 是 迈 过 一 个 门 
楼 而 不 是 目的 ， 真 正 的 较量 是 在 入 职 后 的 成 长 。 惑 像 学 钓鱼 ， 你 可 能 在 
有 经 验 的 垂钓 者 的 指导 下 能 钓 到 几 条 鱼 ， 但 如 果 没 有 学 到 垂钓 的 芮 详 ， 
离开 了 指导 者 你 可 能 就 很 难 钓 到 很 多 鱼 。 我 希望 读 这 本 书 的 朋友 不 要 只 
学 一 些 技巧 来 对 付 面 试 ， 而 是 通过 学 习 如 何 解 决 面试 中 的 难题 来 提高 
己 的 编程 和 解决 问题 的 能 力 ， 进 而 提高 自信 心 ， 在 职场 中 能 迅速 成 长 。 
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Principal Development Manager, Search Technology Center Asia 
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推荐 序 二 


I had the privilege of working with Harry at Microsoft. His background 
and industry experience are a great asset in learning about the process and 
techniques of technical interviews. Harry shares practical information about 
what to expect in a technical interview that goes beyond the core engineering 
skills. An interview is more than a skills assessment. It is the chance for you 
and a prospective employer to gauge whether there is a mutual fit. Harry 
includes reminders about the key factors that can determine a successful 
interview as well as success in your new job. 

Harry takes you through a set of interview questions to share his insight 
into the key aspects of the question. By understanding these questions, you 
can learn how to approach any question more effectively. The basics of 
languages, algorithms and data structures are discussed as well as questions 
that explore how to write robust solutions after breaking down problems into 
manageable pieces. Harry also includes examples to focus on modeling and 
creative problem solving. 

The skills that Harry teaches for problem solving can help you with your 
next interview and in your next job. Understanding better the key problem 
solving techniques that are analyzed in an interview can help you get the first 
job after university or make your next career move. 


Matt Gibbs 
Direct of Development, Asia Research & Development 


Microsoft Corporation 
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2011 年 9 月 份 以 来 ， 我 的 面试 题 博客 
Chttp://zhedahht.blog.163.com/) 点 击 率 上 升 很 快 ， 累 计 点 击 量 超过 了 70 
万 ， 并 且 和 平均 每 天 还 会 增加 约 3000 次 点 击 。 每 一 年 随 着 秋季 新 学 期 的 开 
始 ， 新 一 轮 招 聘 高 峰 也 即将 来 到 。 这 不 禁 让 我 想起 几 年 前 自己 找 工作 的 
情形 。 那 个 时 候 的 我 ， 也 是 在 网 络 的 各 个 角 沙 搜索 面试 经 验 ， 尽 可 能 
地 收集 各 个 公司 的 面试 题 。 

当时 网 上 的 面试 经 验 还 很 和 零散， 应 聘 者 如 果 想 系统 地 收集 面试 题 ， 
需要 付出 很 大 的 努力 。 于 是 我 戎 生 了 一 个 念头 ， 在 博客 上 系统 地 收集 、 
整理 有 代表 性 的 面试 题 ， 这 样 可 以 极 大 地 方便 后 来 人 。 经 过 一 段 时 间 的 
E E E a 


在 过 去 4 年 多 的 日 子 里 ， 我 陆续 发 表 了 60 余 篇 关于 面试 题 的 博客 。 
随 着 博客 数目 的 增加 ， 我 也 逐渐 意识 到 一 篇 篇 博客 仍然 是 零散 的 。 一 篇 
博客 只 是 单纯 地 分 析 一 个 面试 题 ， 但 对 解 题 思路 缺乏 系统 性 的 梳理 。 于 
是 在 2010 年 10 月 我 有 了 把 博客 整理 成 一 本 书 的 想法 。 经 过 一 年 的 努力 ， 
这 本 书 终于 和 读者 见面 了 。 


本 L 内 a 


全 书 分 为 7 章 ， 各 章 的 主要 内 容 如 下 : 

第 1 章 介 绍 面试 的 流程 。 通 常 整个 面试 过 程 可 以 分 为 电话 面试 、 共 
享 桌 面 远程 面试 和 现场 面试 3 个 阶段 ， 每 一 轮 面 试 又 可 以 分 为 行为 面 
试 、 技 术 面 试 和 应 聘 者 提问 3 个 环节 。 本 章 详 细 讨 论 了 面试 中 每 一 环节 
需要 注意 的 问题 。 其 中 第 1.3.2 节 深入 讨论 了 技术 面试 中 的 5 个 要 素 ， 是 
全 书 的 大 纲 ， 接 下 来 的 第 2 一 6 章 逐 一 讨论 每 个 要 点 。 

第 2 章 梳理 应 聘 者 接受 技术 面试 时 需要 用 到 的 基础 知识 。 本 章 从 编 
程 语言 、 数 据 结构 及 算法 三 方面 总 结 了 程序 员 面 试 的 知识 点 。 

第 3 章 讨 论 应聘 者 在 面试 时 写 出 高 质量 代码 的 3 个 要 点 。 通 常 面试 官 
除了 期 得 应聘 者 写 出 的 代码 能 够 完成 基本 的 功能 之 外 ， 还 能 应 对 特殊 情 
况 并 对 非法 输入 进行 合理 的 处 理 。 读 完 这 一 章 ， 读 者 将 学 会 如 何 从 规范 
性 、 完 整 性 和 和 鲁 棒 性 3 个 方面 提高 代码 的 质量 。 

第 4 章 总 结 在 编程 面试 中 解决 难题 的 常用 思路 。 如 果 在 面试 过 程 中 
遇 到 复杂 的 难题 ， 应 聘 者 最 好 在 写 代码 之 前 形成 清晰 的 思路 。 读 者 在 读 



































完 这 一 章 之 后 将 学 会 如 何 用 画图 、 学 例 和 分 解 复 杂 门 是 3 种 电路 来 解 站 
问题 。 

第 5 章 介 绍 如何 优 化 代码 的 时 间 效 率 和 空间 效率 。 如 果 一 个 问题 有 
多 种 解法 ， 面 试 官 总 是 期 竺 应 聘 者 能 找到 最 优 的 解法 。 读 完 这 一 章 ， 读 
者 将 学 会 优化 时 间 效 率 及 空间 换 时 间 的 常用 算法 。 

第 6 章 总 结 面试 中 的 各 项 能 力 。 面 试 官 在 面试 过 程 中 会 一 直 关 注 应 
聘 者 的 学 习 能 力 和 沟通 能 力 。 除 此 之 外 ， 有 些 面 试 官 还 喜欢 考 碍 应 聘 者 
的 知识 迁移 能 力 、 抽 象 建 模 能 力 和 发 散 思 维 能 力 。 读 完 这 一 章 ， 读 者 将 
学 会 如 何 培养 和 运用 这 些 能 力 。 

第 7 章 是 两 个 面试 的 案例 。 在 这 两 个 案例 中 ， 我 们 将 看 到 应 聘 者 在 
面试 过 程 中 的 哪些 举动 是 不 好 的 行为 ， 而 哪些 表现 又 是 面试 官 所 期 得 的 
行为 。 训 心地 希望 应 聘 者 能 在 面试 时 少 犯 甚至 不 犯错 误 ， 完 美 地 表现 出 
自己 的 综合 素质 ， 最 终 拿 到 心仪 的 Offer。 


本 书 特色 


正如 前 面 提 到 的 那样 ， 本 书 的 原型 是 我 过 去 4 年 多 陆 陆 续 续 发 表 的 
几 十 篇 博客 ， 但 这 本 书 也 不 仅仅 是 这 些 博 客 的 总 和 ， 它 在 博客 的 基础 上 
添加 了 如 下 内 容 。 

本 书 试图 以 面试 官 的 视角 来 剖析 面试 题 。 本 书 前 6 章 的 第 一 节 都 
是 “面试 官 谈 面 试 ?”， 收 录 了 分 布 在 不 同 IT 企业 或 者 IT 部 门 ) 的 面试 官 
们 对 代码 质量 、 应 聘 者 如 何 形 成 及 表达 解 题 思路 等 方面 的 理解 。 在 本 书 
中 穿插 着 几 十 条 “面试 小 提示 ”， 是 我 作为 面试 官 给 应 聘 者 在 面试 方法 、 
技巧 方面 的 建议 。 在 第 7 章 的 案例 中 , “面试 官 心理 ”揭示 了 面试 官 在 听 
到 应 聘 者 不 同 回答 时 的 心理 活动 。 应 聘 者 如 果 能 了 解 面试 官 的 心理 活 
动 ， 无 疑 能 在 面试 时 更 好 地 表现 自己 。 

本 书 总 结 了 解决 面试 难题 的 常用 方法 ， 而 不 仅仅 只 是 解决 一 道道 
零散 的 题目 。 在 仔细 分 析 、 解 决 了 几 十 道 典 型 的 面试 题 之 后 ， 我 发 现 其 
实 是 有 一 些 通用 的 方法 可 以 在 面试 的 时 候 帮 助 我 们 解 题 的 。 举 个 例子 ， 
如 果 面 试 的 时 候 遇 到 的 题目 很 难 ， 我 们 可 以 试图 把 一 个 大 的 复杂 的 问题 
分 解 成 若干 个 小 的 简单 的 子 问 题 ， 然 后 递归 地 去 解决 这 些 子 问题 。 再 比 
如 ， 我 们 可 以 用 数组 实现 一 个 简单 的 哈 希 表 解决 一 系列 与 字符 串 相 关 的 
面试 题 。 在 详细 分 析 了 一 道 面 试题 之 后 ， 很 多 章节 都 会 在 “相关 题目 ”中 
列举 出 同类 型 的 面试 题 ， 并 在 “举一反三 ”中 总 结 解决 这 一 类 型 题目 的 方 
法 和 要 点 。 

本 书 收集 的 面试 题 是 都 是 各 大 公司 的 编程 面试 题 ， 极 具 实 战 意 
义 。 包 括 谷 歌 、 微 软 在 内 的 知名 IT 企业 在 招聘 的 时 候 ， 都 非常 重视 应 聘 
者 的 编程 能 力 ， 编 程 技术 面试 也 是 整个 面试 流程 中 最 为 重要 的 一 个 环 
























































节 。 本 书 选取 的 题目 都 是 被 各 大 公司 面试 官 反 复 采 用 的 编程 题 。 如 果 读 
者 一 开始 觉得 书 中 的 有 些 题目 比较 难 ， 那 也 正常 ， 没 有 必要 感到 气 包 -， 
因为 像 谷 歌 、 微 软 这 样 的 大 企业 的 面试 本 喘 束 不 简单 。 读 者 逐步 掌握 了 
书 中 总 结 的 解 题 方法 之 后 ， 编 程 能 力 和 分 析 复 杂 问 题 的 能 力 将 会 得 到 很 
大 的 提升 ， 再 去 大 公司 面试 将 会 轻松 很 多 。 

本 书 附 融 提 供 了 50 道 编程 题 的 完整 的 源 代 码 ， 其 中 包含 了 每 着 题 
的 测试 用 例 。 很 多 面试 官 在 应 聘 者 写 完 程 序 之 后 ， 痢 会 要 求 应 聘 痢 自己 
想 一 些 测试 用 例 来 测试 目 己 的 代码 ， 一 些 没 有 实际 项 目 开 发 经 验 的 应 聘 
者 不 知道 如 何 做 单元 测试 。 相 信 读 者 朋友 在 读 完 这 本 书 之 后 殊 会 知道 如 
何 从 基本 功能 测试 、 边 界 值 测试 、 性 能 测试 等 方面 去 设计 测试 用 例 ， 从 
而 提高 编写 高 质量 代码 的 能 力 。 

本 书 体 例 

在 本 书 的 正文 中 间或 者 章节 的 末尾 ， 罕 插 了 不 少 特殊 体例 。 这 些 体 
T 
ER 

面试 小 提示 : 

本 条 目 是 从 面试 官 的 角度 对 应 聘 者 的 建议 或 者 希望 应 聘 者 能 够 注意 到 的 细节 。 

源 代码 : 


读者 将 在 本 条 目 中 看 到 一 个 格式 为 XX_YYYYY 或 者 XX_Y_ZZZZZ 
的 项 目 名 称 ， 该 名 称 与 用 Visual Studio 打 开 文 件 InterviewQuestions.sln 之 
后 看 到 的 项 目 名称 对 应 。 本 书 附 珊 的 源 代 码 请 到 电子 工业 出 版 社 的 官方 
网 站 下 载 。 读 者 下 载 源 代码 并 解压 缩 之 后 ， 请 用 Visual Studio 2008 或 者 
更 新 的 版 本 阅读 或 者 运行 代码 。 

测试 用 例 : 


本 条 目 列举 应 聘 者 在 面试 时 可 以 用 来 测试 代码 是 否 完整 、 鲁 棒 的 单 
元 测试 用 例 。 通 常 本 书 从 基本 功能 、 边 界 值 、 无 效 的 输入 等 方面 测试 代 
码 的 完整 性 和 重 棱 性 ， 针 对 在 时 间 效 率 或 者 空间 效率 有 要 求 的 面试 题 还 
包含 性 能 测试 的 测试 用 例 。 


本 题 考点 : 
本 条 目 总 络 面 试 官 采用 一 道 面 试题 的 考查 要 点 。 
相关 题目 : 



























































本 条 目 列举 一 些 和 详细 分 析 的 面试 例题 相关 或 者 类 似 的 面试 题 。 
举一反三 ， 


本 条 目 从 解决 面试 例题 中 提炼 出 常用 的 解 题 方 法 。 这 些 解 题 方法 能 够 应 用 到 解决 其 他 同 
型 的 问题 中 去 ， 达 到 举一反三 的 目的 。 


面试 官 心理 : 
在 第 七 章 的 面试 案例 中 ， 本 条 目 用 来 模拟 面试 官 听 到 应 聘 者 的 回答 之 后 的 心理 活动 。 















































Petal Oe, 再 加 上 笔者 的 能 力 有 限 ， 书 中 难免 会 有 一 些 遗 漏 。 
今后 一 旦 发 现 遗 漏 的 问题 ， 我 将 第 一 时 间 在 博客 
Chttp:/zhedahht.blog.163.com/) 上 公布 勤 误 信息 。 读 者 如 果 发 现任 何 问 
题 或 者 有 任何 建议 ， 也 请 在 博客 上 留言 、 评 论 ， 或 者 通过 微 博 
(http://weibo.com/zhedahht) 和 我 联系 。 
致谢 

在 写 博客 及 把 博客 整理 成 书 的 过 程 中 ， 我 得 到 了 很 多 人 的 帮助 。 没 
Sr a a wt ee tee orate re 
谢 ! 

首先 我 要 谢谢 个 人 博客 上 的 读者 。 网 友 们 的 鼓励 让 我 在 博客 上 的 写 
作 从 2007 年 2 月 开始 坚持 到 了 现在 。 也 正 是 由 于 网 友 们 的 就 励 ， 我 最 终 
下 定 决 心 把 博客 整理 成 一 本 书 。 

在 本 书 的 写作 过 程 中 ， 我 得 到 了 很 多 同学 、 同 事 的 帮忙 ， 包 括 
Autodesk) vey. x. EVRY, SC SEAR, AEKA, SKE 
By, Intel, 20uAR TINA, VASE MEPR AK. H 
t, NVidia RXR, SAPHI SEAR AI RAR EBH S VED Beth 
还 在 盛大 工作 ) 。 感 谢 他 们 和 大 家 分 享 了 对 编程 面试 的 理解 和 思考 。 同 
时 还 要 感谢 GlaxoSmithKline Investment 的 Recruitment & HRIS Manager 
咏 来 (也 是 2008 年 把 我 招 进 微软 的 HR〉 和 大 家 分 享 了 微软 所 推崇 的 
STAR 人 简历 模型 。 还 要 感谢 在 微软 期 间 我 的 两 个 老板 徐 鹏 阳 和 Matt 
Gibbs， 他 们 都 是 在 微软 有 十 几 年 面试 经 验 的 资深 面试 官 ， 对 面试 有 着 
a ht A A 
少 。 

我 同样 要 感谢 现在 思科 的 老板 Min Lu 及 TQSG 上 海 团 队 的 同事 王 
观 、 赵 斌 和 朱 波 对 我 的 理解 。 他 们 在 我 写作 期 间 奉 我 分 担 了 大 量 的 工 
作 ， 让 我 能 够 集中 更 多 的 精力 来 写 书 。 

感谢 电子 工业 出 版 社 的 工作 人 员 ， 尤 其 是 张 春 雨 和 赵 树 刚 的 帮助 。 





























两 位 编辑 大 到 全 书 的 构架 ， 小 到 文字 的 推 殴 ， 都 给 予 了 我 极 大 的 帮助 ， 
从 而 使 本 书 的 质量 有 了 极 大 的 提升 。 

本 书 还 得 到 了 很 多 朋友 的 支持 和 帮助 ， 限 于 篇 幅 ， 虽 然 不 能 在 此 一 
一 说 出 他 们 的 和 名字， 但 我 一 样 对 他 们 心 存 感激 。 

最 后 ， 我 要 更 心地 感谢 我 的 爱人 刘 素 云 。 感 谢 她 在 过 去 一 年 中 对 我 
的 理解 和 支持 ， 为 我 营造 了 一 个 温 记 而 又 浪漫 的 家 ， 让 我 能 够 心 无 旁 合 
地 写 书 。 我 无 以 为 谢 ， 说 以 此 书 献 给 她 及 我 们 尚未 出 生 的 小 宝宝 。 

















何 海 涛 
2011 年 9 月 8 日 清晨 于 上 海 三 泾 南 宅 


























let MARITE 
11 面试 官 谈 面 试 


“对 于 初级 程序 员 ， 我 一 般 会 偏向 考查 算法 和 数据 结构 ， 看 应 聘 者 的 基本 功 ， 对 于 高 级 程 
序 员 ， 我 会 多 关注 专业 技能 和 项 目 经 验 。” 












































何 幸 杰 (SAP， 高 级 工程 师 ) 


“应 聘 者 要 事先 做 好 准备 ， 对 公司 近况 、 项 目 情况 有 所 了 解 ， 对 所 应 聘 的 工作 真 的 很 有 热 
情 。 男 外 ， 应 聘 者 还 要 准备 好 合适 的 问题 问 面试 官 。” 







































































一 一 韩 伟 东 《〈 盛 大， 高 级 研究 员 ) 

“应 聘 者 在 面试 过 程 首 先 需要 放松 ， 不 要 过 于 紧张 ， 这 有 助 于 后 面 解决 问题 时 开拓 思路 。 

其 次 不 要 急于 编写 代码 ， 应 该 先 了 解 清楚 所 要 解决 的 问题 。 这 时 候 最 好 先 和 面试 官 多 做 沟通 ， 

然后 开始 做 一 些 整 体 的 设计 和 规划 ， 这 有 助 于 编写 高 质量 和 高 可 读 性 的 代码 。 写 完 代 码 后 不 要 
马上 提交 ， 最 好 自己 review 并 借助 一 些 测试 用 例 来 走 几 通 代码 ， 找 出 可 能 出 现 的 错误 。” 




















































































































马 ?都 是 浮云 ， 应 聘 技术 岗位 就 是 要 踏实 写 程序 。” 


a 
A 
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— HHE (M$, SDE II) 
12 面试 的 三 种 形式 


如 果 应 聘 者 能 够 通过 公司 的 简历 利 选 环节 ， 那 恭喜 他 取得 了 阶段 性 
的 成 功 。 但 要 想 拿 到 心仪 的 Offer， 应 聘 者 还 有 更 长 的 路 要 走 。 大 部 分 公 
司 的 面试 都 是 从 电话 面试 开始 的 。 通 过 电话 面试 之 后 ， 有 些 公 司 还 会 有 
一 两 轮 远程 面试 。 面 试 官 让 应 聘 者 共享 自己 的 加 面 ， 远 程 观察 应 聘 者 编 
写 及 调试 代码 的 过 程 。 如 果 前 面 的 面试 都 很 顺利 ， 应 聘 者 就 会 收 到 现场 
面试 的 邀请 信 ， 请 他 去 公司 接受 面对面 的 面试 。 整 个 面试 的 流程 我 们 可 
以 用 图 1.1 表 示 。 

















现场 面试 






图 1.1 面试 的 形式 和 流程 
TE: 只 有 少数 公司 有 共 孚 桌面 远程 面试 环节 。 




















1.2.1 电话 面试 


顾名思义 ， 电 话 面 试 是 面试 官 以 打 电 话 的 形式 考查 应 聘 者 。 有 些 面 
试 官 会 完 和 应 聘 者 预约 好 电话 面试 的 时 间 ， 而 还 有 些 面试 官 却 喜欢 搞 突 
然 袭 击 ， 一 个 电话 打 过 去 就 开始 面试 。 为 了 应 付 这 种 突然 袭击 ， 建 议 应 
聘 者 在 投 出 简历 之 后 的 一 两 个 星期 之 内 ， 要 保证 手机 电池 能 人 至少 连 续 通 
话 一 个 小 时 。 男 外 ， 应 聘 者 不 要 长 时 间 示 在 很 嘲 杂 的 地 方 。 如 果 应 聘 者 
且 在 亲 市 的 时 候 突然 接 到 面试 电话 ， 那 么 双方 就 有 可 能 因为 听 不 清 对 方 
ITD Fe JE lt JT, 

电话 面试 和 现场 面试 最 大 的 区 别 就 是 应 聘 者 和 面试 官 古 见 不 到 对 方 
的 ， 因 此 双方 的 沟通 只 能 依靠 声音 。 没 有 了 上 肢体 语言 、 面 部 表情 ， 应 聘 
者 清楚 地 表达 自己 想法 的 难度 就 比 现场 面试 时 要 大 很 多 ， 特 别 是 在 解释 
复杂 算法 的 时 候 。 应 聘 者 在 电话 面试 的 时 候 应 尽 可 能 用 形象 化 的 语言 把 
细 世 说 清楚 。 例 如 ， 在 现场 面试 的 时 候 ， 应 聘 者 如 采 想 说 一 个 二 又 树 的 
结构 ， 可 以 用 笔 在 白 纸 上 画 出 来 ， 就 一 目 了 然 。 但 在 电话 面试 的 时 候 ， 
应 聘 者 就 需要 把 二 又 树 中 有 哪些 结 点 ， 每 个 结 点 的 左 子 结 点 是 什么 、 石 
a a 
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很 多 外 企 在 电话 面试 时 都 会 加 上 瑞 语 面试 的 环节 ， 甚 至 有 些 公司 全 
部 面试 都 会 用 英语 进行 。 电 话 面试 时 应 聘 者 只 能 听 到 面试 官 的 声音 而 看 
不 到 他 的 口 型 ， 这 对 应 聘 者 的 听力 提出 了 更 高 的 要 求 。 如 果 应 聘 者 在 面 
试 的 时 候 没 有 听 清 楚 或 者 听 懂 面试 官 的 问题 ， 千 万 不 要 不 懂 装 懂 、 答 非 
所 问 ， 这 是 面试 的 大 忌 。 当 不 确定 面试 官 的 问题 的 时 候 ， 应 聘 者 一 定 要 
大 胆 地 向 面试 官 多 提问 ， 和 直到 和 弄 清楚 面试 官 的 意图 为 止 。 


面试 小 提示 : 


应 聘 者 在 电话 面试 的 时 候 应 尽 可 能 用 形象 的 语言 把 细节 说 清楚 。 
如 果 在 英语 面试 时 没有 听 清 或 没有 上 听 懂 面试 官 的 问题 ， 应 聘 者 要 敢于 说 Pardon。 


1.2.2 ”共享 果 面 远程 面试 


共享 桌面 远程 面试 (Phone-Screen Interview) 是 指 利 用 一 些 共享 桌 
面 的 软件 〈 比 如 微软 的 Live Meeting、 思 科 的 WebEx 等 ) ， 应 聘 者 把 自 
己 电 脑 的 桌面 共享 给 远程 的 面试 官 。 这 样 两 个 人 虽然 没有 坐 在 一 起 ， 但 
面试 官 却 能 通过 共享 保 面 观看 应 聘 者 编程 和 调试 的 过 程 。 目 前 只 有 为 数 
不 多 的 几 家 大 公司 会 在 邀请 应 聘 者 到 公司 参加 现场 面试 之 前 ， 先 进行 一 
两 轮 共享 加 面 的 远程 面试 。 




































































这 种 形式 的 面试 ， 面 试 官 最 关心 的 是 应 聘 者 的 编程 习惯 及 调试 能 
力 。 通 名 面试 官 会 认可 应 聘 者 下 列 几 种 编程 习惯 : 

e 思考 清楚 再 开始 编码 。 应 聘 者 不 要 一 听 到 题目 承 匆 忙 打 开 编 程 
软件 如 Visual Studio 开 始 融 代码 ， 因 为 在 没有 形成 清晰 的 思路 之 前 写 出 
的 代码 通常 会 漏 润 百 出 。 这 些 漏洞 被 面试 官 发 现 之 后 ， 应 聘 者 容易 刁 
张 ， 这 个 时 候 再 修改 代码 也 会 越 改 越 乱 ， 最 终 导 致 面试 的 结果 不 理想 。 
更 好 的 策略 是 应 聘 者 应 先 想 清楚 解决 问题 的 思路 ， 算 法 的 时 间 、 空 间 复 
杂 度 各 是 什么 ， 有 哪些 特殊 情况 需要 处 理 等 ， 然 后 再 动手 编写 代码 。 

e 民 好 的 代码 命名 和 缩 进 对 齐 习 惯 。 一 目 了 然 的 变量 和 函数 名 ， 
的 缩 进 和 括号 对 齐 ， 会 让 面试 官 觉 得 应 聘 者 有 参与 大 型 项 目的 

之 经验 o 

e 能 够 单元 测试 。 通 各 面试 官 出 的 题目 都 是 要 求 写 函数 解决 某 一 
问题 ， 如 果 应 聘 者 能 够 在 定义 函数 之 后 ， 立 即 对 该 函数 进行 全 面 的 单元 
测试 ， 那 束 相 当 于 同 面试 官 证 明了 自己 有 着 专业 的 软件 开发 经 验 。 如 果 
应 聘 者 是 先 写 单 元 测试 用 例 ， 再 写 解决 问题 的 函数 ， 我 相信 面试 官 完 会 
对 你 刮目相看 ， 因 为 能 做 到 测试 在 前 、 开 发 在 后 的 程序 员 实 在 是 太 稀 缺 
了 ， 他 会 毫 不 犹豫 地 抛 出 绿色 的 橄榄 枝 。 

通常 我 们 在 写 代 码 的 时 候 都 会 遇 到 问题 。 当 应 聘 者 运行 代码 发 现 结 
果 不 对 之 后 的 表现 ， 也 是 面试 官 关注 的 重点 ， 因 为 应 聘 者 此 时 的 反应 、 
采取 的 措施 都 能 体现 出 他 的 调试 功底 。 如 果 应 聘 者 能 够 熟练 地 设置 断 
点 、 单 步 跟 踪 、 碍 看 内 存 、 分 析 调 用 栈 ， 能 很 快 发 现 问题 的 根源 并 最 终 
解决 问题 ， 那 么 面试 官 将 会 觉得 他 的 开发 经 验 很 丰富 。 调 试 能 力 是 在 书 
本 上 学 不 到 的 ， 只 有 通过 大 量 的 软件 开发 实践 才能 积累 出 调试 技巧 。 当 
面试 官 发 现 一 个 应 聘 者 的 调试 功底 很 扎实 的 时 候 ， 他 在 写 面 试 报告 的 时 
候 是 不 会 吝 音 赞美 之 词 的 。 


面试 小 提示 : 
在 共享 桌面 远程 面试 过 程 中 ， 面 试 官 最 关心 的 是 应 聘 者 的 编程 习惯 及 调试 能 力 。 


1.2.3 现场 面试 


在 通过 电话 面试 和 共享 加 面 远程 面试 之 后 ， 应 聘 者 不 久 束 会 收 到 E- 
mail， 邀 请 他 去 公司 参加 现场 面试 (Onsite Interview) 。 

去 公司 参加 现场 面试 之 前 ， 应 聘 者 应 做 好 以 下 几 点 准备 : 

o 规划 好 路 线 并 估算 出 行 时 间 。 应 聘 者 要 事先 估算 在 路 上 需要 人 花 
费 多 长 时 间 ， 并 预 留 半 小 时 左右 的 缓冲 时 间 以 应 对 堵车 等 意外 情况 。 如 
果 面 试 运 到 ， 那 至 少 印象 分 会 大 打折 扣 。 






















































































e 准备 好 得 体 的 衣服 。IT 公 司 通常 衣 独 比较 随意 ， 应 聘 者 通 利 没 
有 必要 穿着 正装 ， 一 般 和 舒服 干净 的 衣服 都 可 以 。 

e 注意 面试 邀请 函 里 的 面试 流程 。 如 果 面 试 有 好 几 轮 ， 时 间 也 很 
长 ， 那 么 你 在 面试 过 程 中 可 能 会 觉得 疫 区 并 思维 变 得 迟钝 。 比 如 微软 对 
技术 职位 通常 有 五 轮 面试 ， 连 续 几 个 小 时 处 在 高 压 的 面试 之 中 ， 人 难免 
会 变 得 精 疲 力 尽 。 因 此 应 聘 者 可 以 禹 一 些 提神 的 饮料 或 者 食品 ， 在 两 轮 
面试 之 间 提 神 醒 脑 。 

e 准备 几 个 问题 。 每 一 轮 面试 的 最 后 ， 面 试 官 都 会 让 应 聘 者 问 几 
个 问题 ， 应 聘 者 可 以 提前 准备 好 问题 。 

现场 面试 是 整个 面试 流程 中 的 重头 戏 。 由 于 是 坐 在 面试 官 的 对 面 ， 
应 聘 者 的 一 举 一 动 都 看 在 面试 官 的 眼 里 。 面 试 官 通过 应 聘 者 的 语言 和 行 
动 ， 考 查 他 的 沟通 能 力 、 学 习 能 力 、 编 程 能 力 等 综合 实力 。 本 书 接 下 来 
的 章节 将 详细 讨论 各 种 能 


1.3 ”面试 的 三 个 环 市 


通常 面试 官 会 把 每 一 轮 面试 分 为 三 个 环节 (如 图 1.2 所 示 )〉 : 首先 
是 行为 面试 ， 面 试 冒 参照 简历 了 解 应 聘 者 的 过 往 经 验 ;， 然后 是 技术 面 
试 ， 这 一 坏 市 很 有 可 能 会 要 求 应 聘 者 现场 写 代码 ;最 后 一 个 坏 市 是 应 聘 
者 问 儿 个 自己 最 感 兴趣 的 问题 。 下 面 将 详细 讨论 面试 的 这 三 个 环节 。 


图 1.2 ”面试 的 三 个 环节 
1.3.1 行为 面试 环 市 


面试 开始 的 5 一 10 分 钟 通常 是 行为 面试 的 时 间 。 在 行为 面试 这 个 环 
节 里 ， 面 试 官 会 注意 应 聘 者 的 性 格 特点 ， 深 入 地 了 解 简历 中 列举 的 项 目 
经 历 。 由 于 这 一 环节 一 般 不 会 问 技术 难题 ， 因 此 也 是 一 个 暖 场 的 过 程 ， 
应 聘 者 可 以 利用 这 几 分 钟 时 间 调 整 目 己 的 情绪 ， 进 入 面试 的 状态 。 

不 少 面试 官 会 让 应 聘 者 做 一 个 简短 的 自我 介绍 。 由 于 面试 官 手中 拿 
大 应 聘 者 的 简历 ， 而 那里 有 应 聘 者 的 详细 信息 ， 因 此 此 时 的 目 我 介绍 不 
用 花 很 多 时 间 ， 用 30 秒 到 1 分 钟 的 时 间 介 绍 目 己 的 主要 学 习 、 工 作 经 历 
就 即 可 。 如 果 面 试 官 对 你 的 茶 一 段 经 历 或 者 参与 的 菏 一 个 项 目 很 感 兴 
趣 ， 他 会 有 针对 性 地 提 几 个 问题 详细 了 解 。 
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1. 应 聘 者 的 项 目 经 验 


应 聘 者 日 我 介绍 之 后 ， 面 试 官 接着 会 对 照应 聘 者 的 简历 去 详细 了 解 
他 感 兴趣 的 项 目 。 应 聘 者 在 准备 简历 的 时 候 ， 建 议 用 如 图 1.3 所 示 的 
STAR 模型 描述 自己 经 历 过 的 每 一 个 项 目 。 





为 完成 任务 做 了 


哪些 工作 ， 怎 么 
做 的 





图 1.3 简历 中 描述 项 目的 STAR 模型 


e Situation: 简短 的 项 目 背景 ， 比 如 项 目的 规模 ， 开 发 的 软件 的 
功能 、 目 标 用 户 等 。 

e Task: 自己 完成 的 任务 。 这 个 要 写 详细 ， 要 让 面试 官 对 自己 的 
工作 一 日 了 然 。 在 用 词 上 要 注意 区 分 “参与 "和 “人 负责”， 如 果 只 是 加 入 某 
一 个 开发 团队 写 了 几 行 代码 就 用 “负责 ”"， 那 束 很 危险 。 面 试 官 看 到 简历 
上 应 聘 者 “负责 > 了 某 个 项 目 ， 他 可 能 就 会 问 项 目的 总 体 框架 设计 、 核 心 
算法 、 团 队 合 作 等 问题 。 这 些 问 题 对 于 只 是 简单 “参与 ?的 人 来 说 ， 是 很 
难 回答 的 ， 会 让 面试 官 认为 你 不 诚实 ， 印 象 分 会 减 去 很 多 。 

e Action: 为 了 完成 任务 目 己 做 了 哪些 工作 ， 是 怎么 做 的 。 这 里 
可 以 详细 介绍 。 做 系统 设计 的 ， 可 以 介绍 系统 架构 的 特点 ; 做 软件 开发 
的 ， 可 以 写 基 于 什么 工具 在 哪个 平台 下 应 用 了 哪些 技术 ; 做 软件 测试 
的 ， 可 以 写 是 手工 测试 还 是 自动 化 测试 ， 是 和 白 盒 测试 还 是 黑 盒 测 试 等 。 

e Result: 目 己 的 贡献 。 这 方面 的 信息 可 以 写 得 具体 些 ， 最 好 能 
用 数字 加 以 说 明 。 如 果 是 参与 功能 开发 ， 可 以 说 按时 完成 了 多 少 功能 ; 
如 果 做 优化 ， 可 以 说 性 能 提高 的 百分比 是 多 少 ， 如 果 是 维护 ， 可 以 说 修 
改 了 多 少 个 Bug。 

. 举 个 例子 ， 笔 者 用 下 面 一 段 话 介绍 自己 在 微软 Winforms 项 目 组 的 经 
H: 


”Winforms 是 微软 .NET 中 的 一 个 成 熟 的 UI 平台 (Situation) 。 本 人 的 
工作 是 在 添加 少量 新 功能 之 外 主要 负责 维护 已 有 的 功能 (Task) 。 新 的 

















功能 主要 是 让 Winforms 的 控件 的 风格 和 Vista、Windows 7 的 风格 保持 一 
致 。 在 维护 方面 ， 对 于 较 难 的 问题 我 用 WinDbg 等 工具 进行 调试 
(Action) 。 在 过 去 两 年 中 我 总 共 修 改 了 超过 200 个 Bug (Result) 。 

如 果 在 应 聘 者 的 简历 中 上 述 4 类 信息 还 不 够 清晰 ， 面 试 官 可 能 会 奶 
问 相 关 的 问题 。 除 此 之 外 ， 面 试 官 针对 项 目 经 验 最 常 问 的 问题 还 包括 如 
下 几 个 类 型 : 

e 你 在 该 项 目 中 碰 到 的 最 大 的 问题 是 什么 ， 你 是 怎么 解雇 的 ? 

© 从 这 个 项 目 中 你 学 到 了 什么 ? 

e 什么 时 候 会 和 其 他 团队 成 员 〈 包 括 开 发 人 员 、 测 斌 人员、 设计 
人 员 、 项 目 经 理 等 ) 有 什么 样 的 冲突 ， 你 们 是 怎么 解决 冲突 的 ? 

应 聘 者 在 准备 简历 的 时 候 ， 针 对 每 一 个 项 目 经 历 都 应 提前 做 好 相应 
人 


面试 小 提示 : 


在 介绍 项 目 经 验 〈 包 括 在 简历 上 介绍 和 面试 时 口头 介绍 ) 时 ， 应 聘 者 不 必 详 述 项 目的 背 
景 ， 而 要 突出 介绍 自己 完成 的 工作 及 取得 的 成 绩 。 


2. 应 聘 者 掌握 的 技能 


除了 应 聘 者 参与 过 的 项 目 之 外 ， 面 试 官 对 应 聘 者 掌握 的 技能 也 很 感 
兴趣 ， 他 有 可 能 针对 简历 上 提 到 的 技能 提出 问题 。 和 描述 项 目 时 要 注 
意 “ 参 与 "和 “人 负责” 一样 ， 描 述 技能 掌握 程度 时 也 要 注意 “了 解 "”、“ 熟 
色 ”" 和 “精通 ”的 区 别 。 

“了 解 ” 指 对 某 一 个 技术 只 是 上 过 诬 或 者 看 过 书 ， 但 没有 做 过 实际 的 
项 目 。 通 常 不 建议 在 简历 中 列 出 只 是 肤浅 地 了 解 一 点 的 技能 ， 除 非 这 项 
技术 应 聘 的 职位 的 确 需要 。 比 如 茶 学 生 读本 科 的 时 候 学 过 《计算 机 图 形 
学 》 这 门 课程 ， 但 一 直 没 有 开发 过 与 图 形 绘 制 相关 的 项 目 ， 那 就 只 能 算 
是 了 解 。 如 果 他 去 应 聘 Autodesk 公 司 ， 那 他 可 以 在 简历 上 提 一 下 他 了 解 
图 形 学 。Autodesk 是 一 个 开发 三 维 设计 软件 的 公司 ， 有 很 多 职位 或 多 或 
少 都 会 与 图 形 学 有 关系 ， 那 么 了 解 图形 学 的 总 比 完全 不 了 解 的 要 适合 一 
些 。 但 如 果 他 是 去 应 聘 Oracle， 那 就 没有 必要 提 这 一 点 了 ， 因 为 开发 数 
据 库 系统 的 Oracle 公 司 大 部 分 职位 与 图 形 学 没有 什么 关系 。 

简历 中 我 们 描述 技能 的 掌握 程度 大 部 分 应 该 是 “熟悉 "”。 如 果 我 们 在 
实际 项 目 中 使 用 茶 一 项 技术 已 经 有 较 长 的 时 间 ， 通 过 查阅 相关 的 文档 可 
以 独立 解决 大 部 分 问题 ， 我 们 就 熟悉 它 了 。 对 应 届 毕 业 生 而 言 ， 他 毕业 
设计 所 用 到 的 技能 ， 可 以 用 “熟悉 *， 对 已 经 工作 过 的 ， 在 项 目 开发 过 程 
中 所 用 到 的 技能 ， 也 可 以 用 “熟悉 ”。 






























































如 果 我 们 对 一 项 技术 使 用 得 得 心 应 手 ， 在 项 目 开发 过 程 中 当 同 学 或 
同事 向 我 们 请 教 这 个 领域 的 问题 我 们 都 有 信心 也 有 能 力 解决 ， 这 个 时 候 
我 们 就 可 以 说 自己 精通 了 这 项 拉 术 。 应 聘 者 不 要 试图 在 简历 中 把 自己 修 
饰 成 < 高 人 ?而 轻易 使 用 “精通 ”， 除 非 上 自己 能 够 很 轻松 地 回答 这 个 领域 里 
的 绝 大 多 数 问题 ， 否 则 就 会 适得其反 。 通 常 如果 应 聘 者 在 简历 中 说 上 自己 
精通 某 一 项 技术 ， 面 试 官 就 会 对 他 有 很 高 的 期 望 值 ， 因 此 会 挑 一 些 比 较 
难 的 问题 来 问 。 这 也 是 越 装 高 手 就 越 容易 露馅 的 原因 。 曾 经 碰 到 一 个 在 
简历 中 说 自己 精通 C++ 的 应 聘 者 ， 连 成 员 变 量 的 初始 化 顺序 这 样 的 问题 
都 被 问 得 一 头 雾 水 ， 那 最 终 的 结果 也 就 可 想 而 知 了 。 


3， 回 答 “ 为 什么 跳 档 ” 


在 面试 已 丝 有 工作 经 验 的 应 聘 者 的 时 候 ， 面 试 官 总 喜欢 问 为 什么 打 
算 跳槽 。 每 个 人 都 有 上 自己 的 跳槽 动机 和 原因 ， 因 此 面试 官 也 不 会 期 竺 一 
个 标准 答案 。 面 试 官 只 是 想 通 过 这 个 问题 来 了 解 应 聘 者 的 性 格 ， 因 此 应 
聘 者 可 以 大 胆 地 根据 自己 的 真实 想法 来 回答 这 个 问题 。 但 是 ， 应 聘 者 也 
不 要 想 说 什么 就 说 什么 ， 以 免 给 面试 官 留 下 负面 的 印象 。 

在 回答 这 个 问题 时 不 要 抱 优 ， 也 不 要 流露 出 负面 的 情绪 。 负 面 的 情 
绪 通 常 是 能 够 传染 的 ， 当 应 聘 者 总 是 在 抱 候 的 时 候 ， 面 试 官 就 会 担心 如 
末 把 他 招 进来 的 话 他 将 成 为 团队 负面 情绪 的 传染 源 ， 从 而 影响 整个 团队 
的 士气 。 应 聘 者 应 尽量 避免 以 下 4 个 原因 : 

e 老板 太 苛 刻 。 如 末 面 试 官 就 是 当前 招聘 的 职位 的 老板 ， 他 听 到 
应 聘 者 抱怨 现在 的 老板 苛刻 时 ， 他 肯定 会 想 要 是 把 这 个 人 招 进来 ， 接 下 
来 他 就 会 抱怨 我 也 苛刻 了 。 

e 同事 太 难 相处 。 如 果 应 聘 者 说 他 周围 有 很 多 很 难 相处 的 同事 ， 
面试 官 很 有 可 能 会 觉得 这 个 人 他 本 吴 就 很 难 相处 。 

e 加 班 太 频繁 。 对 于 大 部 分 开 企 业 来 说 ， 加 班 是 家 常 便 饭 。 如 果 
正在 面试 的 公司 也 需要 经 常 加 班 ， 那 等 于 应 聘 者 说 他 不 想 进 这 家 公司 。 

e 工资 太 低 。 现 在 的 工资 太 低 的 确 是 大 部 分 人 跳槽 的 真实 原因 ， 
但 不 建议 在 面试 的 时 候 对 面试 官 抱怨 。 面 试 的 目的 是 拿 到 offer， 我 们 要 
尽量 给 面试 官 留 下 好 印象 。 现 在 假设 你 是 面试 官 ， 有 两 个 人 来 面试 : 一 
个 人 一 开口 就 说 现在 工资 太 低 了 ， 和 希望 新 工作 能 加 多 少 多 少 工资 ， 妃 一 
个 次 我 只 管 努 力 干 活 ， 工 资 公司 看 着 给 ， 相 信 公 司 不 会 亏 待 勤 备 的 员 
工 。 你 更 喜欢 哪个 ? 这 里 不 是 说 工资 不 重要 ， 但 我 们 要 清楚 面试 不 是 谈 
工资 的 时 候 。 等 完成 技术 面试 之 后 谈 offer 的 时 候 ， 再 和 HR 谈 工资 也 不 
人 迟 。 通 过 面试 之 后 我 们 就 掌握 主动 了 ， 想 怎么 谈 就 怎么 谈 ， 如 果 工 资 真 
的 开 高 了 HR 会 和 你 很 客气 地 商量 。 









































笔者 在 面试 的 时 候 ， 通 各 给 出 的 答案 是 : 现在 的 工作 做 了 一 段 时 
间 ， 已 经 没有 太 多 的 激情 了 ， 因 此 和 希望 寻找 一 份 更 有 挑战 的 工作 。 然 后 
具体 论述 为 什么 有 些 厌 倦 现 在 的 职位 ， 以 及 面试 的 职位 我 为 什么 会 有 兴 
趣 。 笔 者 目 己 跳 过 两 次 模 ， 第 一 次 从 Autodesk 跳 槽 到 微软 ， 第 二 次 从 微 
软 跳槽 到 现在 的 思科 。 从 面试 的 结果 来 看 ， 这 样 的 回答 都 让 面试 官 很 满 
意 ， 最 终 也 都 拿 到 了 offer。 

当时 在 微软 面试 被 问 到 为 什么 要 跳槽 时 ， 笔 者 的 回答 是 : 我 在 
Autodesk 开 发 的 软件 Civil ” 3D 是 一 球面 同 土木 行业 的 设计 软件 。 如 果 我 
想 在 现在 的 职位 上 得 到 提升 ， 融 必须 加 强 土 木 行业 的 学 习 ， 可 我 对 诸如 
计算 土方 量 、 道 路 设计 等 没有 太 多 兴趣 ， 因 此 出 来 寻找 机 会 。 

在 微软 工作 两 年 半 之 后 去 思科 面试 的 时 候 ， 笔 者 的 回答 是 : REM 
软 的 主要 工作 是 开发 和 维护 .NET 的 UI 平 台 Winforms。 由 于 Winforms 已 
经 非 党 成熟， 不 需要 添加 多 少 新 功能 ， 因 此 我 的 大 部 分 工作 都 是 维护 和 
修改 BUG。 两 年 下 来 ， 调 试 的 能 力 得 到 了 很 大 的 提高 ， 但 长 期 如 此 上 自己 
的 软件 开发 和 设计 能 力 将 不 能 得 到 提高 ， 因 此 想 出 来 寻找 可 以 设计 和 开 
发 系统 的 职位 。 同 时 ， 我 在 过 去 几 年 里 的 工作 都 是 开发 昌 面 软件 ， 对 网 
络 了 解 其 少 ， 因 此 和 希望 下 一 个 工作 能 与 网 络 相 关 。 众 所 周知 ， 思 科 是 个 
ee 
只 位 很 感 兴 趣 。 


1.3.2 技术 面试 环 市 


面试 官 在 通过 简历 及 行为 面试 大 臻 了 解 应 聘 者 的 背景 之 后 ， 接 下 来 
就 要 开始 扩 术 面试 了 。 一 轮 1 小 时 的 面试 ， 通 常 技术 面试 会 占据 40 一 50 
分 钟 。 这 是 面试 的 重头 戏 ， 对 面试 的 结果 起 决定 性 作用 。 虽 然 不 同 公司 
里 不 同 面试 官 的 背景 、 性 格 各 不 相同 ， 但 总 体 来 说 他 们 都 会 关注 应 聘 者 
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图 1.4 应聘 者 需要 具备 的 素质 
应 聘 者 在 面试 之 前 需要 做 足 准 备 ， 对 编程 语言 、 数 据 结 构 和 算法 等 








基础 知识 有 全 面 的 了 解 。 面 试 的 时 候 如 果 遇 到 简单 的 问题 ， 应 聘 者 一 定 
要 注重 细节 ， 写 出 完整 、 鲁 棒 的 代码 。 如 果 遇 到 复杂 的 问题 ， 应 聘 者 可 
以 通过 画图 、 举 具体 例子 分 析 和 分 解 复 杂 问 题 等 方法 先 理 清 思路 再 动手 
编程 。 除 此 之 外 ， 应 聘 者 还 应 该 不 断 优 化 时 间 效 率 和 空间 效率 ， 力 求 找 
到 最 优 的 解法 。 在 面试 过 程 中 ， 应 聘 者 还 应 该 主动 提问 ， 以 弄 清楚 题目 
的 要 求 ， 表 现 目 己 的 沟通 能 力 。 当 面试 官 前 后 问 的 两 个 问题 有 相关 性 的 
时 候 ， 尽 量 把 解决 前 面 问题 的 思路 迁移 到 后 面 的 问题 中 去 ， 展 示 目 己 民 
好 的 学 习 能 力 。 如 果 能 做 到 这 么 儿 点 ， 那 么 通过 面试 获得 心仪 的 职位 将 
古 水 到 渠 成 的 事情 。 


1. 扎实 的 基础 知识 


扎实 的 基本 功 是 成 为 优秀 程序 员 的 前 提 条 件 ， 因 此 面试 官 首 要 关注 
的 应 聘 者 素质 束 是 是 合 具 备 扎实 的 基础 知识 。 通 党 基本功 在 编程 面试 环 
市 体现 在 3 个 方面 : 编程 语言 、 数 据 结构 和 算法 。 

首先 ， 每 个 程序 员 人 至 少 要 掌握 一 两 门 编程 语言 。 面 试 官 从 应 聘 者 在 
面试 过 程 中 写 的 代码 及 跟 进 的 提问 中 ， 能 看 出 其 编程 语言 掌握 的 玖 练 程 
度 。 以 大 部 分 公司 面试 有 要求 的 C++ 举例 。 如 果 写 的 函数 需要 传 入 一 个 指 
针 ， 面 试 官 可 能 会 问 是 否 需 要 为 该 指针 加 上 const， 把 const 加 在 指针 不 
同 的 位 置 是 否 有 了 区别 ; 如 采写 的 函数 需要 传 入 的 参数 是 一 个 复杂 类 型 的 
实例 ， 面 试 官 可 能 会 问 传 入 值 参 数 和 传 入 引用 参数 有 什么 区 别 ， 什 么 时 
候 需 要 为 传 入 的 引用 参数 加 上 const。 

其 次 ， 数 据 络 构 通常 是 编程 面试 过 程 中 考 碍 的 重点 。 在 参加 面试 之 
前 ， 应 聘 者 需要 熟练 掌握 链表 、 树 、 栈 、 队 列 和 哈 希 表 等 数据 结构 ， 以 
及 扎 们 的 操作 。 如 宋 我 们 留意 各 大 公司 的 面试 题 ， 就 会 发 现 链表 和 二 又 


















































树 相 关 的 问题 是 很 多 面试 官 喜 欢 问 的 问题 。 这 方面 的 问题 看 似 比 较 简 
单 ， 但 要 真正 掌握 也 不 容易 ， 特 别 适 合 在 这 么 短 的 面试 时 间 内 检验 应 聘 
者 的 基本 功 。 如 果 应 聘 者 事先 对 链表 的 插入 和 删除 结 点 了 如 指 掌 ， 对 二 
叉 树 的 各 种 吉 历 方法 的 循环 和 递归 写法 者 烂熟 于 胸 ， 那 么 真正 到 了 面试 
的 时 候 也 就 游 力 有 余 了。 

最 后 ， 大 部 分 公司 都 会 注重 考查 人 查找、 排序 等 算法 。 应 聘 者 可 以 在 
了 解 各 种 查找 和 排序 算法 的 基础 上 ， 重 点 掌握 二 分 查找 、 归 并 排序 和 快 
速 排序 ， 因 为 很 多 面试 题 都 只 是 这 些 算法 的 变 体 而 已 。 比 如 面试 题 8“ 旋 
转 数组 的 最 小 数字 ?和 面试 题 38“ 数 字 在 排序 数组 中 出 现 的 次 数 ” 的 本 质 
是 考 碍 二 分 查找 ， 而 面试 题 36“ 数 组 中 的 逆序 对 ”实际 上 是 考 碍 归并 排 
序 。 少 数 对 算法 很 重视 的 公司 比如 谷歌 或 者 百度 ， 还 会 要 求 应 聘 者 熟练 
掌握 动态 规划 和 贫 林 算法。 如果 应 聘 者 对 动态 规划 算法 很 熟悉 ， 那 么 他 
就 能 很 轻松 地 解决 面试 题 31“ 连 续 子 数组 的 最 大 和 ”。 

在 本 书 的 第 2 半 “ 面 试 需要 的 基础 知识 ”中 ， 我 们 将 详细 介绍 应 聘 者 
需要 画 练 掌握 的 基础 知识 。 


2. 高 质量 的 代码 


只 有 注重 质量 的 程序 员 ， 才 能 写 出 鲁 棒 稳 定 的 大 型 软件 。 在 面试 过 
程 中 ， 面 试 官 总 会 格外 关注 边界 条 件 、 特 殊 输 入 等 看 似 细 枝 术 市 但 实质 
至 天 重要 的 地 方 ， 以 考查 应 聘 痢 是 否 注 重 代 码 质 量 。 很 多 时 候 ， 面 试 官 
发 现 应 聘 者 写 出 来 的 代码 只 能 完成 最 基本 的 功能 ， 一 旦 输入 特殊 的 边界 
条 件 参 数 束 会 错误 百出 甚至 程序 朋 江 。 

总 有 些 应 聘 者 很 困惑 : 面试 的 时 候 沉 得 题目 很 简单 ， 感 党 目 己 都 做 
出 来 了 ， 可 最 后 为 什么 锐 拒 了 呢 ? 面试 被 拒 有 很 多 种 可 能 ， 比 如 面试 官 
认为 你 性 格 不 适合 、 态 度 不 够 城 尽 等 。 但 在 技术 面试 过 程 中 ， 这 些 都 不 
古 最 重要 的 。 抠 术 面 试 的 面试 官 一 般 都 是 程序 员 ， 程 序 员 通 向 没有 那么 
多 想法 ， 他 们 只 认 一 个 理 : 题目 做 对 、 做 完整 了 ， 就 让 你 通过 面试 ， 否 
则 失败 。 所 以 遇 到 简单 题目 却 被 拒 的 情况 ， 应 聘 者 应 认真 反思 在 思路 或 
者 代码 中 存在 哪些 漏洞 。 

以 微软 面试 开发 工程 师 时 最 常用 的 一 个 问题 为 例 ; 把 一 个 字符 串 转 
ee 
行 的 代码 : 




































































int StrToInt (char* string) 


{ 


int number = 0; 
while(*string != 0) 
number = number * 10 + *string - '0' 


++string; 


} 
return number; 


看 了 上 面 的 代码 ， 你 是 不 是 党 得 微软 面试 很 容易 ? 如 果 你 真 的 这 人 么 
想 ， 那 你 可 能 又 要 被 拒 了 。 

通常 越 是 简单 的 问题 ， 面 试 官 的 期 望 值 束 会 越 高 。 如 果 题 目 很 简 
单 ， 面 试 官 就 会 期 符 应 聘 者 能 够 很 完整 地 解决 问题 ， 除 了 完成 基本 功能 
之 外 ， 还 要 考 感到 边界 条 件 、 错 误 处 理 等 各 个 方面 。 比 如 这 道 题 ， 面 试 
官 不 仅仅 是 期 竺 你 能 完成 把 字符 串 转 换 成 整数 这 个 最 起 码 的 要 求 ， 而 且 
希望 你 能 考虑 到 各 种 特殊 的 输入 。 面 试 官 衬 少 会 期 待 应 聘 者 能 够 在 不 需 
要 提示 的 情况 下 ， 考 虑 到 输入 的 字符 串 中 有 非 数字 字符 和 正 负 和 号， 要 考 
碟 到 最 大 的 正 整 数 和 最 小 的 负 整数 以 及 流出 。 同 时 面试 官 还 期 竺 应 聘 者 
能 够 考虑 到 当 输 入 的 字符 串 不 能 转换 成 整数 时 ， 应 该 如 何 做 错误 处 理 。 
当 把 这 个 问题 的 方方面面 邦 考 虑 到 的 时 候 ， 我 们 束 不 会 青 认为 这 道 题 简 


单 了 。 

除了 问题 考虑 不 全 面 忆 外， 还 有 一 个 面试 官 不 能 容 恕 的 错误 就 是 程 
序 不 够 鲁 棒 。 以 前 面 的 那 段 代码 为 例 ， 只 要 输入 一 个 空 指针 ， 程 序 立 即 
月 泪 。 这 样 的 代码 如 果 加 入 到 软件 当中 ， 将 是 灾难 。 因 此 当面 试 官 看 到 
代码 中 对 空 指针 没有 判断 并 加 以 特殊 处 理 的 时 候 ， 通 向 他 连 往 下 看 的 兴 
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趣 都 没有 。 
当然 ， 不 是 所 有 与 鲁 棒 性 相关 的 问题 都 和 前 面 的 代码 那样 明显 。 再 


举 一 个 很 多 人 痢 曾 经 被 面试 过 的 问题 ， 求 链表 中 的 倒数 第 k 个 结 点 。 有 
不 少 人 在 面试 之 前 在 网 上 看 过 这 个 题目 ， 因 此 知道 思路 是 用 两 个 指针 ， 
第 一 个 指针 先 走 k 一 1 步 ， 然 后 两 个 指针 一 起 走 。 当 第 一 个 指针 走 到 尾 结 
扩 的 时 候 ， 第 二 个 指针 指向 的 就 是 倒数 第 k 个 结 点 。 于 是 他 大 笔 一 挥 ， 

写 下 了 下 面 的 代码 : 








ListNode* FindKthToTail(ListNode* pListHead, unsigned int k) 
{ 


if (pListHead == NULL) 
return NULL; 


ListNode *pAhead = pListHead; 
ListNode *pBehind = NULL; 


for(unsigned int i = 0; i< k =- 17 ++ 1) 
{ 
pAhead = pAhead->m_pNext; 


} 
pBehind = pListHead; 


while (pAhead->m_pNext != NULL) 
{ 
pAhead = pAhead->m_pNext; 
pBehind = pBehind->m_pNext; 


} 
return pBehind; 


写 完 之 后 ， 应 聘 者 看 到 自己 已 经 判断 了 输入 的 指针 是 不 是 空 指针 并 
做 了 特殊 处 理 ， 于 是 以 为 这 次 面试 必定 能 顺利 通过 ， 可 是 他 没有 想到 的 
是 这 段 代 码 中 仍然 有 很 严重 的 问题 : 当 链 表 中 的 结 点 总 数 小 于 k 的 时 
候 ， 程 序 还 是 会 月 渍 。 另 外 ， 当 输入 的 k 为 0 时 ， 同 样 也 会 引起 程序 裔 
涡 。 因 此 ， 几 天 之 后 他 收 到 的 仍然 不 是 Offer 而 是 拒 信 。 

要 想 很 好 地 解决 前 面 的 问题 ， 最 好 的 办 法 是 在 动手 写 代 码 之 后 想 好 
测试 用 例 。 只 有 把 各 种 可 能 的 输入 事先 都 想 好 了 ， 才 能 在 写 代码 的 时 候 
把 各 种 情况 都 做 相应 的 处 理 。 写 完 代码 之 后 ， 也 不 要 立刻 给 面试 官 检 
查 ， 而 是 先 在 心里 默默 地 运行 。 当 输入 之 前 想 好 的 所 有 测试 用 例 都 能 得 
到 合理 的 输出 时 ， 再 把 代码 交 给 面试 官 。 做 到 了 这 一 步 ， 通 过 面试 拿 到 
Offer 就 是 顺理成章 的 事情 了 。 
人 
9 方法。 


面试 小 提示 : 


面试 官 除 了 希望 应 聘 者 的 代码 能 够 完成 基本 的 功能 之 外 ， 还 会 关注 应 聘 者 是 否 考虑 了 边 
界 条 件 、 特 殊 输 入 〈 比 如 NULL 指 针 ， 空 字符 串 等 ) 及 错误 处 理 。 
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3. 清晰 的 思路 


只 有 思路 清晰 ， 应 聘 者 才 有 可 能 在 面试 过 程 中 解决 复杂 的 问题 。 有 
些 时 候 面 试 官 会 有 意 出 一 些 比较 复杂 的 问题 ， 以 考查 应 聘 者 能 售 在 短 时 
间 内 形成 清晰 的 思路 并 解决 问题 。 对 于 确实 很 复杂 的 问题 ， 面 试 官 甚至 
不 期 竺 应聘 者 能 在 面试 不 到 一 个 小 时 的 时 间 里 给 出 完整 的 答案 ， 他 更 看 
重 的 可 能 还 是 应 聘 者 是 否 有 清晰 的 思路 。 面 试 官 通常 不 喜欢 应 聘 者 在 没 
有 形成 清晰 思路 之 前 就 草率 地 开始 写 代 码 ， 这 样 写 出 来 的 代码 容易 逻辑 
混乱 、 错 误 百 出 。 

应 聘 者 可 以 用 几 个 简单 的 方法 帮助 自己 形成 清晰 的 思路 。 首 先是 举 
儿 个 简单 的 具体 例子 让 自己 理解 问题 。 当 我 们 一 眼看 不 出 问题 中 隐藏 的 
规律 的 时 候 ， 可 以 试 着 用 一 两 个 具体 的 例子 模拟 操作 的 过 程 ， 这 样 说 不 
定 就 能 通过 具体 的 例子 找到 抽象 的 规律 。 其 次 可 以 试 着 用 图 形 表 示 抽 象 
的 数据 结构 。 像 分 析 与 链表 、 二 又 树 相 关 的 题目 ， 我 们 都 可 以 画 出 它们 
的 结构 来 简化 题目 。 最 后 可 以 试 着 把 复杂 的 问题 分 解 成 寿 干 个 简单 的 子 
问题 ， 再 一 一 解决 。 很 多 基于 递归 的 思路 ， 包 括 分 治 法 和 动态 规划 ， 都 
古 把 复 林 的 问题 分 解 成 一 个 或 者 多 个 简单 的 子 问 题 。 

比如 把 二 又 搜 索 树 转换 成 排序 的 双向 链表 这 个 问题 就 很 复杂 。 遇 到 
这 个 问题 ， 我 们 不 妨 先 画 出 一 两 个 具体 的 二 又 搜索 树 ， 直 观 地 感受 二 又 
搜索 树 和 排序 的 双 同 链表 有 哪些 联系 。 如 果 一 下 子 找 不 出 转换 的 规律 ， 
我 们 可 以 把 整个 二 又 树 看 成 3 个 部 分 : 根 结 点 、 左 子 树 和 右 子 树 。 妆 我 
们 递归 地 把 转换 左右 子 树 这 两 个 子 问题 解决 之 后 ， 再 把 转换 元 右 子 树 得 
到 的 链表 和 根 结 点 链接 起 来 ， 整 个 问题 也 就 解决 了 ( 详 见面 试题 27“ 二 
又 搜索 树 与 双 同 链表 ”)。 

在 本 书 的 第 4 草 “ 解 诀 面试 题 的 思路 ?中 ， 我 们 将 详细 讨论 遇 到 复杂 
问题 时 如 何 采 用 画图 、 举 例 和 分 解 问题 等 方法 帮助 我 们 解决 问题 。 


面试 小 提示 : 


如 果 在 面试 的 时 候 遇 到 难题 ， 我 们 有 3 种 办 法 分 析 、 解 决 复杂 的 问题 : 画图 能 使 抽象 问题 
形象 化 ， 举 例 使 抽象 问题 具体 化 ， 分 解 使 复杂 问题 简单 化 。 


4. 优化 效率 的 能 


优秀 的 程序 员 对 时 间 和 内 存 的 消耗 锚 铁 必 较 ， 他 们 很 有 激情 地 不 断 
优化 目 己 的 代码 。 当 面试 官 出 的 题目 有 多 种 解法 的 时 候 ， 通 利他 会 期 符 
应 聘 者 最 终 能 够 找到 最 优 解 。 当 面试 官 提示 还 有 更 好 的 解法 的 时 候 ， 应 
聘 者 不 能 放弃 思考 ， 而 应 该 努力 寻找 在 时 间 消耗 或 者 空间 消耗 上 可 以 优 
















































































化 的 地 方 。 

要 想 优 化 时 间或 者 空间 效率 ， 首 先 要 知道 如 何 分 析 效 率 。 即 使 是 同 
一 个 算法 ， 用 不 同方 法 实现 的 效率 可 能 也 会 大 不 相同 ， 我 们 要 能 够 分 析 
出 算法 及 其 代码 实现 的 效率 。 例 如 求 斐 波 那 契 数列 ， 很 多 人 喜欢 用 递归 
公式 f Cn) =f Cn 一 1) +f Cn 一 2) 求解 。 如 果 分 析 它 的 递归 调用 树 ， 我 
们 就 会 发 现 有 大 量 的 计算 是 重复 的 ， 时 间 复 杂 度 以 n 的 指数 增加 。 但 如 
果 我 们 先 求 f (1) 、f (2) ， 再 根据 f Ga) 和 f (2) 求 出 f (3) ， 接 下 来 
根据 f (2) 、f (3) RHE (4) ， 并 以 此 类 推 用 一 个 循环 求 出 f Cn) ， 
人 


要 想 优化 代码 的 效率 ， 我 们 还 要 熟知 各 种 数据 结构 的 优 缺 点 ， 并 能 
选择 合适 的 数据 结构 解决 问题 。 我 们 在 数组 中 根据 下 标 可 以 用 O (1) 
时 间 完 成 查找 。 数 组 的 这 个 特征 可 以 用 来 实现 简单 的 哈 希 表 解 决 很 多 问 
题 ， 比 如 面试 题 35“ 第 一 个 只 出 现 一 次 的 字符 ”。 为 了 解决 面试 题 30“ 最 
小 的 k 个 数 ”， 我 们 需要 一 个 数据 容器 来 存储 k 个 数字 。 在 这 个 数据 容器 
中 ， 我 们 希望 能 够 快速 地 找到 最 大 值 并 且 能 快速 地 替换 其 中 的 数字 。 经 
过 权衡 ， 我 们 发 现 二 又 树 比如 最 大 堆 或 者 红 黑 树 都 是 实现 这 个 数据 容器 
的 不 错 选 择 。 

要 想 优 化 代码 的 效率 ， 我 们 也 要 熟练 掌握 常用 的 算法 。 面 试 中 最 常 
用 的 算法 是 查找 和 排序 。 如 果 从 头 到 尾 顺序 扫描 一 个 数组 ， 我 们 需要 
O n) 时 间 才 能 完成 查找 操作 。 但 如 果 数 组 是 排序 的 ， 应 用 二 分 查找 
算法 就 能 把 时 间 复 杂 度 降低 到 O Cogn) 〈 如 面试 题 8% 旋 转 数 组 的 最 小 
值 ? 和 面试 题 38“ 数 字 在 排序 数组 中 出 现 的 次 数 ") 。 排 序 算法 除了 能 够 
给 数组 排序 之 外 ， 还 能 用 来 解决 其 他 问题 。 比 如 快速 排序 算法 中 的 
Partition 函数 能 够 用 来 在 n 个 数 里 查找 第 k 大 的 数字 ， 从 而 解雇 面试 题 
29“ 数 组 中 出 现 次 数 超过 一 半 的 数字 ”和 面试 题 30“ 最 小 的 k 个 数 "。 归 并 
排序 算法 能 够 实现 在 O (Cnlogn) 时 间 统 计 n 个 数字 中 的 逆序 对 数目 〈 面 
试题 36“ 数 组 中 的 逆序 对 ”) 。 

在 本 书 的 第 5 章 “ 优 化 时 间 空 间 效 率 ” 中 ， 我 们 将 详细 讨论 如 何 从 时 
间 效 率 和 空间 效率 两 方面 去 做 优化 。 


5. 优秀 的 综合 能 力 


在 面试 过 程 中 ， 应 聘 者 除了 展示 自己 的 编程 能 力 和 技术 功 捅 之 外 ， 
还 需要 展示 自己 的 软 技能 (Soft Skills〉， 诸 如 自己 的 沟通 能 力 和 学 习 能 
力 。 随 着 软件 系统 的 规模 越 来 越 大 ， 软 件 开发 已 经 告别 了 单打 独 斗 的 年 
代 ， 程 序 员 与 他 人 的 沟通 变 得 越 来 越 重 要 。 在 面试 过 程 中 ， 面 试 官 会 观 




















察 应 聘 者 在 介绍 项 目 经 验 或 者 算法 思路 时 是 否 观点 明确 、 逻 辑 清晰 ， 并 
以 此 判断 其 沟通 能 力 的 强 弱 。 另 外 ， 面 试 官 也 会 从 应 聘 者 说 话 的 神态 和 
语气 来 判断 他 是 否 有 团队 合作 的 意识 。 通 常 面试 官 不 会 喜欢 高 做 或 者 轻 
视 合作 者 的 人 。 

IT 行业 知识 更 新 很 快 ， 因 此 程序 员 只 有 具备 很 好 的 学 习 能 力 才能 跟 
上 知识 更 普 的 步伐 。 通 常 面 试 官 有 两 种 办 法 考查 应 聘 者 的 学 习 能 力 。 面 
试 官 的 第 一 种 方法 是 询问 应 聘 者 最 近 在 看 什么 书 、 从 中 学 到 了 哪些 新 技 
术 。 面 试 官 可 以 用 这 个 问题 了 解 应 聘 者 的 学 习 愿 望 和 学 习 能 力 。 面 试 官 
的 第 二 种 方法 是 抛 出 一 个 新 概念 ， 接 下 来 他 会 观察 应 聘 者 能 不 能 在 较 短 
时 间 内 理解 这 个 新 概念 并 解决 相关 的 问题 。 比 如 面试 官 要 求 应 聘 者 计算 
第 1500 个 丑 数 。 很 多 人 都 没有 听 说 过 导数 这 个 概念 。 这 个 时 候 面试 官 就 
会 观察 应 聘 者 面 对 丑 数 这 个 新 概念 时 ， 能 不 能 经 过 提问 、 思 考 、 再 提问 
的 这 程 ， 最 终 找 昌 导数 的 规律 从 而 找到 解决 方案 《主见 面试 是 34 
Ht”) 。 

知识 迁移 能 力 是 一 种 特殊 的 学 习 能 力 。 如 果 我 们 能 够 把 已 经 掌握 的 
知识 迁移 到 其 他 领域 ， 那 么 学 习 新 技术 或 者 解决 新 间 题 就 会 变 得 容易 。 
面试 官 经 常会 先 问 一 个 简单 的 问题 ， 再 问 一 个 很 复杂 但 和 前 面 的 简单 问 
题 相关 的 问题 。 这 个 时 候 面试 官 期 待 应 聘 者 能 够 从 简单 问题 中 得 到 启 
示 ， 从 而 找到 解决 复杂 问题 的 窍门 。 比 如 面试 官 先 要 求 应 聘 者 写 一 个 函 
数 求 斐 波 那 契 数列 ， 再 问 一 个 青蛙 跳台 阶 的 问题 ， 一 只 青蛙 一 次 可 以 中 
上 1 级 台阶 ， 也 可 以 跳 上 2 级 台阶 。 请 问 这 只 青蛙 跳 上 n 级 台阶 总 共有 多 
少 种 跳 法 。 应 聘 者 如 果 具 有 较 强 的 知识 迁移 能 力 ， 就 能 分 析出 青蛙 跳台 
入 问题 实质 上 只 是 小数 列 的 一 个 应 用 CH HR EAE 
IP) 。 

还 有 不 少 面试 官 喜欢 考查 应 聘 者 的 抽象 建 模 能 力 和 发 散 思 维 能 力 。 
面试 官 从 日 常生 活 中 提炼 出 问题 ， 比 如 面试 题 44“ 扑 克 牌 中 的 顺 子 *”， 考 
查 应 聘 者 能 不 能 把 问题 抽象 出 来 用 合理 的 数据 结构 表示 ， 并 找到 其 中 的 
规律 解决 这 个 问题 。 面 试 官 也 可 以 限制 应 聘 者 不 得 使 用 常规 方法 ， 这 要 
求 应 聘 者 具备 创新 精神 ， 能 够 打开 思路 从 多 角度 去 分 析 、 解 决 问题 。 比 
如 在 面试 题 47“ 不 用 加 减 乘除 做 加 法 "中 ， 面 试 官 期 待 应 聘 者 能 够 打开 思 
路 ， 用 位 运算 实现 整数 的 加 法 。 

我 们 将 在 本 书 的 第 6 章 “ 面 试 中 的 各 项 能 力 "中 用 具体 的 面试 题 详细 
讨论 上 述 能 力 在 面试 中 的 重要 作用 。 


1.3.3 ”应 聘 痢 提问 环 市 
在 结束 面试 前 的 5 一 10 分 钟 ， 面 试 官 会 给 应 聘 者 机 会 问 几 个 问题 ， 





















































应 聘 者 的 问题 的 质量 对 面试 的 结果 也 有 一 定 的 影响 。 有 些 人 的 沟通 能 
很 强 ， 马 上 就 能 想到 有 意思 的 问题 。 但 对 于 大 多 数 人 而 言 ， 在 经 受 了 面 
试 官 将 近 一 小 时 的 拷问 之 后 可 能 已 经 精疲力竭 ， 再 迅速 想 出 儿 个 问题 难 
度 很 大 。 因 此 建议 应 聘 者 不 妨 在 面试 之 前 做 些 功 读 ， 为 每 一 轮 面 试 准 备 
2 一 3 个 问题 ， 这 样 到 提问 环节 的 时 候 就 游 妃 有 余 了 。 

面试 官 让 应 聘 者 问 几 个 问题 ， 主 要 是 想 了 解 他 最 关心 的 问题 有 哪 
些 ， 因 此 应 聘 者 全 少 要 问 一 两 个 问题 ， 售 则 面试 官 束 会 觉得 你 对 我 们 公 
司 、 职 位 等 都 不 感 兴趣 ， 那 你 来 面试 做 什么 ? 但 是 也 不 是 什么 问题 都 可 
以 在 这 个 时 候 问 。 如 宁 问 题 问 得 比较 合适 ， 对 应 聘 者 来 说 是 个 加 分 的 好 
机 会 ， 但 如 果 问 的 问题 不 太 合 适 ， 面 试 官 对 他 的 印象 就 会 大 打折 扣 。 

有 些 问 题 是 不 适合 在 技术 面试 这 个 环节 里 问 的 。 首 先是 不 要 问 和 目 
己 的 职位 没有 关系 的 问题 ， 比 如 问 “ 公 司 未 来 五 年 的 发 展 战略 是 什么 ”。 
如 果 应 聘 的 职位 是 CTIO， 而 面试 官 是 CEO， 这 倒是 个 合适 的 问题 。 如 采 
应 聘 的 只 是 在 一 线 开 发 的 职位 ， 那 这 个 问题 离 我 们 就 太 远 了 ， 与 我 们 的 
切 吴 利益 没有 多 少 关 系 。 必 外 ， 坐 在 对 面 的 面试 官 很 有 可 能 也 只 是 一 个 
在 一 线 开发 的 程序 员 ， 他 该 怎么 回答 这 个 关系 公司 发 展 战 略 的 问题 呢 ? 

其 次 是 不 要 问 薪 水 。 技 术 面 试 不 是 谈 薪水 的 时 候 ， 要 谈 工 资 要 等 通 
过 面试 之 后 和 HR 谈 。 而 且 让 面试 官 党 得 你 最 关心 的 问题 就 是 新 水 ， 给 
面试 官 留 下 的 印象 也 不 好 。 

再 次 是 不 要 立即 打听 面试 结果 ， 比 如 问 “ 您 觉得 我 能 拿 到 Offer 吗 ”之 
类 的 问题 。 现 在 大 部 分 公司 的 面试 都 有 好 几 轮 ， 最 终 决 定 应 聘 者 能 不 能 
通过 面试 ， 是 要 把 所 有 面试 官 的 评价 综合 起 来 的 。 问 这 个 问题 相当 于 白 
问 ， 因 为 问 了 面试 官 也 不 可 能 告诉 应 聘 者 结果 ， 还 会 让 面试 官 沉 得 他 没 
有 目 我 评估 的 能 力 。 

最 后 推荐 问 的 问题 是 与 招聘 的 职位 或 者 项 目 相关 的 问题 。 如 果 这 种 
类 型 的 问题 问 得 很 到 位 ， 那 么 面试 官 束 会 觉得 你 对 应 聘 的 职位 很 有 兴 
趣 。 不 过 要 问好 这 种 类 型 的 问题 也 不 容易 ， 因 为 首先 对 应 聘 的 职位 或 者 
项 目的 背景 要 有 一 定 的 了 解 。 我 们 可 以 从 两 方面 去 了 解 相 关 的 信息 : 一 
征 面试 前 做 足 功 读 ， 到 网 上 去 收集 一 些 相关 的 信息 ， 做 到 对 公司 成 立时 
间 、 主 要 业务 、 职 位 要 求 等 都 了 然 于 胸 ， 二 是 面试 过 程 中 留心 面试 官 说 
过 的 话 。 有 不 少 面试 官 在 面试 之 前 会 简单 介绍 与 招聘 职位 相关 的 项 目 ， 
其 中 会 包含 其 他 渠道 无 法 得 到 的 信息 ， 比 如 项 目 进展 情况 每 。 应 聘 者 可 
以 从 中 找 出 一 两 个 点 ， 然 后 向 面试 官 提问 。 

下 面 的 例子 古 笔者 去 思科 面试 时 间 的 几 个 问题 。 一 个 面试 官 介 绍 项 
目 时 说 这 次 招聘 是 项 目 组 第 一 次 在 中 国 招 人 ， 目 前 这 个 项 目 所 有 人 员 都 
在 美国 总 部 。 这 轮 面 试 笔者 最 后 问 的 问题 是 : 这 个 项 目 所 有 的 老 员 工 都 
在 美国 ， 那 怎么 对 中 国 这 一 批 新 员工 进行 培训 ? 中 国 的 新 员工 有 没有 机 


















































会 去 美国 总 部 学 习 ? 最 后 一 轮 面 试 是 老板 面试 ， 她 介绍 说 正在 招聘 的 项 
目 组 负责 开发 一 个 测试 系统 ， 思 科 用 它 来 测试 供应 丙 生 产 的 网 络 设备 。 
这 一 轮 笔 者 问 的 几 个 问题 是 ， 这 个 组 是 做 测试 系统 的 ， 那 这 个 组 的 人 员 
古 不 是 也 要 参与 网 络 设备 的 测试 ?是 不 是 需要 学 习 人 硬件 测试 相关 的 知 

R? 因为 我 们 测试 的 对 象 是 网 络 设备 ， 那 么 这 个 职位 对 网 络 硬件 的 掌握 
程度 有 没有 要 求 ? 


1.4 本 章 小 结 


本 章 重 点 介绍 了 面试 的 流程 。 通 常 面试 是 从 电话 面试 开始 的 。 接 下 
来 可 能 有 一 两 轮 共 孚 桌面 远程 面试 ， 面 试 官 通过 桌面 共 吝 软 件 远 程 考 奋 
应 聘 者 的 编程 和 调试 能 力 。 如 果 应 聘 者 的 表现 足够 优秀 ， 那 么 公司 将 邀 
请 他 到 公司 去 接受 现场 面试 。 

一 般 每 一 轮 面 试 都 有 三 个 环节 。 首 先是 行为 面试 环节 ， 面 试 官 在 这 
一 环节 中 对 照 简历 询问 应 聘 者 的 项 目 经 验 和 掌握 的 技能 。 接 下 来 就 是 拉 
术 面 试 环节 ， 这 是 面试 的 重头 戏 。 在 这 一 环节 里 ， 面 试 官 除了 关注 应 聘 
者 的 编程 能 力 和 技术 功底 之 外 ， 还 会 注意 考查 他 的 沟通 能 力 和 学 习 能 
人 
I 问题 。 

本 章 的 1.3.2 市 是 全 书 的 大 纲 。 本 节 介 绍 了 面试 官 天 注 应 聘 者 5 个 方 
面 的 素质 ;基础 知识 是 否 扎实 、 能 人 否 写 出 高 质量 的 代码 、 思 路 是 否 清 
晰 、 和 是否 有 优化 效率 的 能 力 ， 以 及 包括 学 习 能 力 、 沟 通 能 力 在 内 的 综合 
素质 是 否 优秀 。 在 接 下 来 的 第 2 章 到 第 6 草 中 我 们 将 一 一 深入 探讨 这 5 个 
方面 的 素质 。 



































He ”面试 需要 的 基础 知识 


2.1 面试 官 谈 基础 知识 


“C++ 的 基础 知识 ， 如 面向 对 象 的 特性 、 构 造 函 数 、 析 构 函 数 、 动 态 绑 定 等 ， 能 够 反映 出 

应 聘 者 是 否 善 于 把 握 问 题 本 质 ， 有 没有 耐心 深入 一 个 问题 。 另 外 还 有 常用 的 设计 模式 、UML 图 

等 ， 这 些 都 能 体现 应 聘 者 是 否 有 软件 工程 方面 的 经 验 。” 

一 ”王海波 (Autodesk， 软 件 工 程 师 ) 

“对 基础 知识 的 考查 我 特别 重视 C++ 中 对 内 存 的 使 用 管理 。 我 觉得 内 存 管理 是 C++ 程序 员 
特别 要 注意 的 ， 因 为 内 存 的 使 用 和 管理 会 影响 程序 的 效率 和 稳定 性 。 

—— 蓝 诚 (Autodesk， 软 件 工 程 师 ) 


“基础 知识 反映 了 一 个 人 的 基本 能 力 和 基础 素质 ， 是 以 后 工作 中 最 核心 的 能 力 要 求 。 我 一 
ices : (1) E (2) 编程 能 力 ; G) 部 分 数学 知识 ， 如 概率 ;〈4) 问题 的 
a 和 推理 能 v 


























































































































































































































— ik RA CARE, RAED 


“我 比较 重视 四 块 基 础 知识 : O 编程 基本 功 《〈 特 别 喜 欢 字 符 串 处 理 这 一 类 的 问题 ) 
(2) 并 发 控制 ，(3) 算法 、 复 杂 度 ; (O 语言 的 基本 概念 。” 


一 一 张 瑞 百度， 高 级 软件 工程 师 ) 
“我 会 考查 编程 基础 、 计 算 机 系统 基础 知识 、 算 法 以 及 设计 能 力 。 这 些 是 一 个 软件 工程 师 

的 最 基本 的 东西 ， 这 些 方面 表现 出 色 的 人 ， 我 们 一 般 认 为 是 有 发 展 潜力 的 。” 
一 一 韩 伟 东 〈 盛 大， 高 级 研究 员 ) 


“ (1) 对 0OS 的 理解 程度 。 这 些 知 识 对 于 工作 中 第 遇 到 的 内 存 管 理 、 文 件 操作 、 程 序 性 

、 多 线程 、 程 序 安 全 等 有 重要 帮助 。 对 于 OS 理解 比较 深入 的 人 对 于 偏 底层 的 工作 上 手 一 般 比 

公认 (2) 对 于 一 门 编程 语言 的 掌握 程度 。 一 个 热爱 编程 的 人 应 该 会 对 某 种 语言 有 比较 深入 的 

了 解 。 通常 这 样 的 人 对 于 新 的 编程 语言 上 手 也 比较 快 ， 而 且 理 解 比较 深入 。 (3) 常用 的 算法 和 
数据 结构 。 不 了 解 这 些 的 程序 员 基 本 只 能 写 写 ‘Hello World’. 















































































































































































































































WAZ (微软 ，SDE I) 
2.2 ”编程 语言 


程序 员 写 代码 总 是 基于 茶 一 种 编程 语言 ， 因 此 技术 面试 的 时 候 直 接 

或 者 间接 都 会 涉及 至 少 一 种 编程 语言 。 在 面试 的 过 程 中 ， 面 试 官 要 么 直 
言 的 语法 ， 要 么 让 应 聘 者 用 一 种 编程 语言 写 代 码 解 决 一 个 问题 ， 

通过 写 出 的 代码 来 判断 应 聘 者 对 他 使 用 的 语言 的 掌握 程度 。 现 在 流行 的 

















编程 语言 很 多 ， 不 同 公司 开 发 用 的 语言 也 不 上 尽 相 同 。 做 底层 开发 比如 经 
常 写 驱 动 的 人 更 习惯 用 C，Linux 下 有 很 多 程序 员 用 C++ 开 发 应 用 程序 ， 
基于 Windows 的 C# 项 目 己 经 越 来 越 多 ， 路 平台 开发 的 程序 员 则 可 能 更 喜 
欢 Java， 随 着 鞋 果 iPad、iPhone 的 热 销 已 经 有 很 多 程序 员 投 向 了 
Objective “C 的 阵营 ， 同 时 还 有 很 多 人 喜欢 用 脚本 语言 如 Perl、Python 开 
发 短小 精致 的 小 应 用 软件 。 因 此 ， 不 同 公 司 面 试 的 时 候 对 编程 语言 的 要 
求 也 有 所 不 同 。 每 一 种 编程 语言 都 可 以 写 出 一 本 大 部 头 的 书籍 ， 本 书 限 
于 篇 幅 不 可 能 面面俱到 。 本 书 中 所 有 代码 都 用 C/C++/C# 实 现 ， 下 面 简要 
介绍 一 些 C++/C# 常 见 的 面试 题 。 


Zl, CHF 


国内 绝 大 部 分 高 校 都 开设 C++ 的 课程 ， 因 此 绝 大 部 分 程序 员 都 学 过 
C++， 于 是 C++ 成 了 各 公司 面试 的 首选 编程 语言 。 包 括 Autodesk 在 内 的 
很 多 公司 在 面试 的 时 候 会 有 大 量 的 C++ 的 语法 题 ， 其 他 公司 虽然 不 直接 
面试 C++ 的 语法 ， 但 面试 题 要 求 用 C++ 实现 算法 。 因 此 总 的 说 来 ， 应 聘 
者 不 管 去 什么 公司 求职 ， 都 应 该 在 一 定 程度 上 和 掌握 C++。 

通常 语言 面试 有 3 种 类 型 。 第 一 种 类 型 是 面试 官 直接 询问 应 聘 者 对 
C++ 概念 的 理解 。 这 种 类 型 的 问题 ， 面 试 官 特别 喜欢 了 解 应 聘 者 对 
C++ 关键 字 的 理解 程度 。 例 如 : 在 C++ 中 ， 有 哪 4 个 与 类 型 转换 相关 的 关 
键 字 ? 这些 关 键 字 各 有 什么 特点 ， 应 该 在 什么 场合 下 使 用 ? 

在 这 种 类 型 的 题目 中 ，sizeof 是 经 常 被 问 到 的 一 个 概念 。 比 如 下 面 
的 面试 片段 ， 就 反复 出 现在 各 公司 的 技术 面试 中 。 

































































面试 官 : 定义 一 个 空 的 类 型 ， 
到 的 结果 是 多 少 ? 
应 聘 者 : 答案 是 1。 
面试 官 : 为 什么 不 是 0? 
应 聘 者 : 空 类 型 的 实例 中 不 包含 任何 信息 ， 本 来 求 sizeof 应 该 是 0， 但 是 当 我 们 声明 该 类 
型 的 实例 的 时 候 ， 它 必须 在 内 存 中 占有 一 定 的 空间 ， 和 否则 无 法 使 用 这 些 实 例 。 至 于 占用 多 少 内 
存 ， 由 编译 器 决定 。Visual Studio 中 每 个 空 类 型 的 实例 占用 1 字 节 的 空间 。 
面试 官 : 如 果 在 该 类 型 中 添加 一 个 构造 函数 和 析 构 函数 ， 再 对 该 类 型 求 sizeof， 得 到 的 结 
果 又 是 多 少 ? 
应 聘 者 ;和 前 面 一 样 ， 还 是 1。 调 用 构造 函数 和 析 构 函数 只 需要 知道 函数 的 地 址 即 可 ， 而 
这 些 函 数 的 地 址 只 与 类 型 相关 ， 而 与 类 型 的 实例 无 关 ， 编 译 器 也 不 会 因为 这 两 个 函数 而 在 实例 
内 添加 任何 额外 的 信息 。 
面试 官 ， 那 如 果 把 析 构 函数 标记 为 虚 函 数 呢 ? 
应 聘 者 : ++ 的 编译 器 一 旦 发 现 一 个 类 型 中 有 虚拟 函数 ， 就 会 为 该 类 型 生成 虚 函 数 表 ， 并 


里 面 没有 任何 成 员 变 量 和 成 员 函 数 。 对 该 类 型 求 sizeof， 得 













































































































































































在 该 类 型 的 每 一 个 实例 中 添加 一 个 指向 虚 函 数 表 的 指针 。 在 32 位 的 机 器 上 ， 一 个 指针 占 4 字 节 的 
空间 ， 因 此 求 sizeof 得 到 4， 如 果 是 64 位 的 机 器 ， 一 个 指针 占 8 字 节 的 空间 ， 因 此 求 sizeof 则 得 到 
8。 











面试 C/C++ 的 第 二 种 题 型 就 是 面试 官 拿 出 事先 准备 好 的 代码 ， 让 应 
聘 者 分 析 代 人 码 的 运行 结果 。 这 种 题 型 选择 的 代码 通常 包含 比较 复杂 微妙 
的 语言 特性 ， 这 要 求 应 聘 者 对 C++ 考点 有 着 透彻 的 理解 。 即 使 应 聘 者 对 
eR 
Fe gh Lz 

比如 面试 官 递 给 应 聘 者 一 张 有 如 下 代码 的 A4 打 印 纸 要 求 他 分 析 编 
译 运 行 的 结果 ， 并 提供 3 个 选项 : A. 编译 错误 ;B. 编 译 成 功 ， 运 行 时 程 
Feria; C. 编 译 运行 正常 ， 输 出 10。 








A(A other) { value = other.value; } 


void Print() { std::cout << value << std::endl; } 


int tmain(int argc, TCHAR* argv[]) 


A 
A 
b.Pr 
return 0; 


在 上 述 代 码 中 ， 复 制 构造 函数 A (A other) 传 入 的 参数 是 A 的 一 个 
实例 。 由 于 是 传 值 参数 ， 我 们 把 形 参 复制 到 实 参 会 调用 复制 构造 函数 。 
因此 如 果 人 允许 复 制 构造 函数 传 值 ， 就 会 在 复制 构造 函数 内 调用 复制 构造 
函数 ， 就 会 形成 永 无 休止 的 递归 调用 从 而 导致 栈 洲 出 。 因 此 C++ 的 标准 
不 允许 复制 构造 函数 传 值 参数 ， 在 Visual Studio 和 GCC 中 ， 都 将 编译 出 
错 。 要 解雇 这 个 问题 ， 我 们 可 以 把 构造 函数 修改 为 A〈const 
A&other) ， 也 就 是 把 传 值 参 数 改 成 常量 引用 。 

第 三 种 题 型 就 是 要 求 应 聘 者 写 代 码 定义 一 个 类 型 或 者 实现 类 型 中 的 
成 员 函 数 。 让 应 聘 者 写 代 码 的 难度 自然 比 让 应 聘 者 分 析 代 码 要 高 不 少 ， 
因为 能 想 明 白 的 未 必 就 能 写 得 清楚 。 很 多 考查 C++ 语法 的 代码 题 围绕 在 
构造 函数 、 析 构 函 数 及 运算 符 重 载 。 比 如 面试 题 1“ 赋 值 运 算 符 函 数 ? 就 
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是 一 个 例子 。 

为 了 让 大 家 能 顺利 地 通过 C++ 面试 ， 更 重要 的 是 能 更 好 地 学 习 党 握 
C++ 这 门 编程 语言 ， 这 里 推荐 几 本 C++ 的 书 ， 大 家 可 以 根据 自己 的 具体 
情况 选择 阅读 的 顺序 : 

e (Effective C++》。 这 本 书 很 适合 在 面试 之 前 突击 C++。 这 本 
书 列举 了 使 用 C++ 经 常 出 现 的 问题 及 解决 这 些 问 题 的 技巧 。 该 书 中 提 到 
的 问题 也 是 面试 官 很 喜欢 问 的 问题 。 

e 《C++ Primer》。 访 完 这 本 书 ， 就 会 对 C++ 的 语法 有 全 面 的 了 
解 。 

e 《Inside C++ Object Model) 。 这 本 书 有 助 于 我 们 深入 了 解 
C++ 对 象 的 内 部 。 读 懂 这 本 书后 很 多 C++ 难题 ， 比 如 前 面 的 sizeof 的 问 
题 、 虚 函数 的 调用 机 制 等 ， 都 会 变 得 很 容易 。 

e <The C++ Programming Language) 。 如 果 是 想 全 面 深 入 掌握 
C++， 没 有 哪 本 书 比 这 本 书 更 适合 的 了 。 


面试 题 1， 赋 值 运算 符 函 数 
题 日 :如 下 为 类 型 CMyString 的 声明 ， 请 为 该 类 型 添加 赋 
值 运算 符 函 数 。 
M 


s CMyString 




















public: 
CMyString(char* pData = NULL); 
CMyString (const CMyStringé str); 
~CMyString (void); 


当面 试 官 要 求 应 聘 者 定义 一 个 赋值 运算 符 函 数 时 ， 他 会 在 检查 应 聘 
者 写 出 的 代码 时 关注 如 下 几 点 : 

e 是 否 把 返回 值 的 类 型 声明 为 该 类 型 的 引用 ， 并 在 函数 结束 前 返 
回 实例 自身 的 引用 《〈 即 this) 。 只 有 返回 一 个 引用 ， 才 可 以 允许 连续 赋 
值 。 否 则 如 果 函 数 的 返回 值 是 void， 应 用 该 赋值 运算 符 将 不 能 做 连续 赋 
值 。 假 设 有 3 个 CMyString 的 对 象 : str1、str2 和 str3， 在 程序 中 语句 
str1=str2=str3 将 不 能 通过 编译 。 

e 是否 把 传 入 的 参数 的 类 型 声明 为 常量 引用 。 如 果 传 入 的 参数 不 
是 引用 而 是 实例 ， 那 么 从 形 参 到 实 参 会 调用 一 次 复制 构造 函数 。 把 参数 





声明 为 引用 可 以 避免 这 样 的 无 谓 消 耗 ， 能 提高 代码 的 效率 。 同 时 ， 我 们 
在 赋值 运算 符 函 数 内 不 会 改变 传 入 的 实例 的 状态 ， 因 此 应 该 为 传 入 的 引 
用 参数 加 上 const 关 键 字 。 

e 是 否 释 放 实 例 自 身 己 有 的 内 存 。 如 果 我 们 环 记 在 分 配 新 内 存 之 
前 释放 目 身 己 有 的 空间 ， 程 序 将 出 现 内 存 泄 露 。 

e。 是 否 判断 传 入 的 参数 和 当前 的 实例 Cthis) 是 不 是 同一 个 实 
例 。 如 果 是 同一 个 ， 则 不 进行 厂 值 操作 ， 直 接 返 回 。 如 果 事 先 不 判断 惑 
进行 赋值 ， 那 么 在 释放 实例 自嘲 的 内 存 的 时 候 束 会 导致 严 重 的 问题 : 
当 *this 和 传 入 的 参数 是 同一 个 实例 时 ， 那 么 一 旦 释放 了 自 员 的 内 存 ， 传 
入 的 参数 的 内 存 也 同时 被 释放 了 ， 因 此 再 也 找 不 到 需要 赋值 的 内 容 了 。 


经 典 的 解法 ， 适 用 于 初级 程序 员 
当 我 们 完整 地 考虑 了 上 述 4 个 方面 之 后 ， 我 们 可 以 写 出 如 下 的 代 











CMyStringé CMyString::operator =(const CMyString éstr) 
{ 
if (this == éstr) 
return *this; 


delete [lm pData; 

m pData = NULL; 

m pData = new char[strlen(str.m pData) + 1]; 
strcpy(m pData, str.m pData); 


return *this; 


这 是 一 般 C++ 教 材 上 提供 的 参考 代码 。 如 果 接 受 面 试 的 是 应 届 毕 业 
生 或 者 C++ 初 级 程序 员 ， 能 全 面 地 考虑 到 前 面 四 点 并 完整 地 写 出 代码 ， 
面试 官 可 能 会 让 他 通过 这 轮 面 试 。 但 如 果 面 试 的 是 C++ 局 级 程序 员 ， 面 
试 官 可 能 会 提出 更 高 的 要 求 。 


考虑 异常 安全 性 的 解法 ， 局 级 程序 员 必 备 


在 前 面 的 函数 中 ， 我 们 在 分 配 内 存 之 前 先 用 delete 释 放 了 实例 
m_pData 的 内 存 。 如 果 此 时 内 存 不 足 导 致 new char 抛 出 异常 ，m_pData 将 


日 

是 

符 函 数 内 部 抛 出 一 个 异常 ，CMyString 的 实例 不 再 保持 有 效 的 状态 ， 这 
就 违背 了 异常 安全 性 (Exception Safety) 原则 。 


} 











要 想 在 赋值 运算 符 冰 数 中 实现 异常 安全 性 ， 我 们 有 两 种 方法 。 一 个 
简单 的 办 法 是 我 们 先 用 new 分 配 新 内 容 再 用 delete 释 放 已 有 的 内 容 。 这 样 
只 在 分 配 内 容 成 功 之 后 再 释放 原来 的 内 容 ， 也 束 是 当 分 配 内 存 失败 时 我 
们 能 确保 CMyString 的 实例 不 会 被 修改 。 我 们 还 有 一 个 更 好 的 办 法 是 先 
创建 一 个 临时 实例 ， 再 交换 临时 实例 和 原来 的 实例 。 下 面 是 这 种 思路 的 
参考 代码 : 

CMyString& CMyString::operator =(const CMyString &str) 
{ 


if(this != &str) 
CMyString strTemp (str); 


char* pTemp = strTemp.m_pData; 
strTemp.m_pData = m_pData; 
m_pData = pTemp; 


return *this; 
} 

在 这 个 函数 中 ， 我 们 先 创 建 一 个 临时 实例 strTemp， 接 着 把 
strTemp.m_pData 和 实例 自身 的 m_pData 做 交换 。 由 于 strTemp 是 一 个 局 
部 变量 ， 但 程序 运行 到 证 的 外 面 时 也 就 出 了 该 变量 的 作用 域 ， 束 会 上 自动 
调用 strTemp 的 析 构 函数 ， 把 strTemp.m_pData 所 指向 的 内 存 释 放 掉 。 由 
于 strTemp.m_pData 指 同 的 内 存 就 是 实例 之 前 m_pData 的 内 存 ， 这 就 相当 
于 目 动 调用 析 构 函数 释放 实例 的 内 存 。 

在 新 的 代码 中 ， 我 们 在 CMyString 的 构造 函数 里 用 new 分 配 内 存 。 如 
果 由 于 内 存 不 足 抛 出 诸如 bad_alloc 等 异常 ， 我 们 还 没有 修改 原来 实例 的 
状态 ， 因 此 实例 的 状态 还 是 有 效 的 ， 这 也 束 保 证 了 异常 安全 性 。 

如 果 应 聘 者 在 面试 的 时 候 能 够 考虑 到 这 个 层面 ， 面 试 官 就 会 党 得 他 
se 的 异常 安全 性 有 很 深 的 理解 ， 那 么 他 上 自然 也 就 能 通过 这 轮 面 试 

















源 代码 : 
本 题 完 整 的 源 代码 详 见 01_AssignmentOperator 项 目 。 
测试 用 例 : 


e 把 一 个 CMyString 的 实例 赋值 给 另外 一 个 实例 。 
e 把 一 个 CMyString 的 实例 赋值 给 它 目 己 。 


e 连续 赋值 。 
本 题 考点 : 


。 考查 对 C++ 的 基础 语法 的 理解 ， 如 运算 符 函数 、 常 量 引用 等 。 
。 考查 对 内 存 泄露 的 理解 。 
mn 对 高 级 C++ 程序 员 ， 面 试 官 还 将 考查 应 聘 者 对 代码 蜡 常安 全 人 
4 理解 。 


2.2.2 C# 


C# 是 微软 在 推出 新 的 开发 平台 .NET 时 同步 推出 的 编程 语言 。 由 于 
Windows 至 今 仍然 是 用 户 最 多 的 操作 系统 ， 而 .NET 义 是 微软 近年 来 力 推 
的 开发 平 侣 ， 因 此 C# 无 论 在 桌面 软件 还 是 网 络 应 用 的 开 及 上 都 有 痢 广泛 
的 应 用 ， 所 以 我 们 也 不 难 理解 为 什么 现在 很 多 基于 Windows 系 统 开发 的 
公司 部 会 要 求 应 聘 者 掌握 C#。 

C# 可 以 看 成 是 一 门 以 C++ 为 基础 发 展 起 来 的 一 种 托管 语言 ， 因 此 它 
的 很 多 关键 字 甚 至 语法 都 和 C++ 很 类 似 。 对 一 个 学 习 过 C++ 编程 的 程序 
员 而 言 ， 他 用 不 了 多 长 时 间 学 习 就 能 用 C# 来 开 及 软件 。 然 而 我 们 也 要 清 
醒 地 认识 到 ， 昌 然 学 习 C# 与 C++ 相同 或 者 类 似 的 部 分 很 容易 ， 但 要 和 掌握 
并 区 分 两 者 不 同 的 地 方 却 不 是 一 件 很 容易 的 事情 。 面 试 官 总 是 喜欢 深 完 
我 们 模棱两可 的 地 方 以 考 碍 我 们 是 不 是 真 的 理解 了 ， 因 此 我 们 要 着 重 注 
意 C# 与 C++ 不 同 的 语法 特点 。 下 面 的 面试 片段 就 是 一 个 例子 : 
面试 官 ， C++ 中 可 以 用 struct 和 class 来 定义 类 型 。 这 两 种 类 型 有 什么 区 别 ? 

应 聘 者 : 如 果 没 有 标明 成 员 函 数 或 者 成 员 变 量 的 访问 权限 级 别 ， 在 struct 中 默认 的 是 
public， 而 在 class 中 默认 的 是 private。 
面试 官 : 那 在 C# 中 呢 ? 

应 聘 者 : C# 和 C++ 不 一 样 。 在 C# 中 如 果 没 有 标明 成 员 函 数 或 者 成 员 变 量 的 访问 权限 级 
别 ，struct 和 class 中 都 是 private 的 。struct 和 class 的 区 别 是 struct 定 义 的 是 值 类 型 ， 值 类 型 的 实例 在 
栈 上 分 配 内 存 ; 而 class 定 义 的 是 引用 类 型 ， 引 用 类 型 的 实例 在 堆 上 分 配 内 存 。 

在 C# 中 ， 每 个 类 型 中 和 C++ 一 样 ， 都 有 构造 函数 。 但 和 C++ 不 同 的 
是 ， 我 们 在 C# 中 可 以 为 类 型 定义 一 个 Finalizer 和 Dispose 方 法 以 释放 资 
源 。Finalizer 方 法 虽然 写法 与 C++ 的 析 构 函数 看 起 来 一 样 ， 都 是 后 面 跟 
类 型 名 字 ， 但 与 C++ 析 构 函数 的 调用 时 机 是 确定 的 不 同 ，C# 的 Finalizer 
古 在 运行 时 〈CLR) 做 垃圾 回收 时 才 会 被 调用 ， 它 的 调用 时 机 是 由 运行 
时 决定 的 ， 因 此 对 程序 员 来 说 是 不 确定 的 。 男 外 ， 在 C# 中 可 以 为 类 型 定 
义 一 个 特殊 的 构造 函数 ， 静 态 构 造 阔 数 。 这 个 函数 的 特点 是 在 类 型 第 一 
次 被 使 用 之 前 由 运行 时 自动 调用 ， 而 且 保证 只 调用 一 次 。 关 于 静态 构造 



































































































































函数 ， 我 们 有 很 多 有 意思 的 面试 题 ， 比 如 运行 下 面 的 C# 代 码 ， 输 出 的 结 
果 是 什么 ? 
— < A 

public A(string text) 

{ 


Console.WriteLine (text); 


class B 
{ 
static A al = new A("al"); 
A a2 = new A("a2"); 
static B() 
{ 
al = new A("a3"); 
} 
public B() 
{ 
a2 = new A("a4") 
} 
} 
class Program 


static void Main(string[] args) 
{ 
B b = new B(); 


} 


在 调用 类 型 B 的 代码 之 前 先 执行 B 的 静态 构造 函数 。 静 态 构造 函数 
先 初 始 化 类 型 的 静态 变量 ， 再 执行 函数 体内 的 语句 。 因 此 先 打印 al 再 打 
印 a3。 接 下 来 执行 Bb=newB © ， 即 调用 B 的 普通 构造 函数 。 构 造 函 数 
先 初始 化 成 员 变 量 ， 再 执行 函数 体内 的 语句 ， 因 此 先后 打印 出 a2、a4。 
因此 运行 上 面 的 代码 ， 得 到 的 结果 将 是 打印 出 4 行 ， 分 别 是 al、a3、 
a2、a4。 

我 们 除了 要 关注 C# 和 C++ 不 同 的 知识 点 之 外 ， 还 要 格外 关注 C# 一 些 
特有 的 功能 ， 比 如 反射 、 应 用 程序 域 CAppDomain) 等 。 这 些 概念 还 相 
互 关 联 ， 要 花 很 多 时 间 学 习 研 究 才 能 透彻 地 理解 它们 。 下 面 的 代码 就 是 
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一 段 关 于 反射 和 应 用 程序 域 的 代码 ， 运 行 它 得 到 的 结果 是 什么 ? 
[Serializable] 
internal class A : MarshalByRefObject 


{ 
public static int Number; 


public void SetNumber(int value) 
{ 
Number = value; 
} 
} 


[Serializable] 
internal class B 


{ 
public static int Number; 


public void SetNumber(int value) 
{ 
Number = value; 
} 
} 


class Program 


{ 
static void Main(string[] args) 
{ 
String assambly = Assembly.GetEntryAssembly() .FullName; 
AppDomain domain = AppDomain.CreateDomain("NewDomain") ; 


A.Number = 10; 

String nameOfA = typeof (A) .FullName; 

A a = domain.CreateInstanceAndUnwrap(assambly, nameOfA) as A; 
a.SetNumber (20); 

Console.WriteLine ("Number in class A is {0}", A.Number); 


B.Number = 10; 

String nameOfB = typeof (B) .FullName; 

B b = domain.CreateInstanceAndUnwrap(assambly, nameOfB) as B; 
b.SetNumber (20) ; 

Console.WriteLine ("Number in class B is {0}", B.Number); 


上 述 C# 代 人 码 先 创建 一 个 名 为 NewDomain 的 应 用 程序 域 ， 并 在 该 域 
中 利用 反射 机 制 创建 类 型 A 的 一 个 实例 和 类 型 B 的 一 个 实例 。 我 们 注意 
到 类 型 A 是 继承 自 MarshalByRefObject， 而 B 不 是 。 虽 然 这 两 个 类 型 的 结 
构 一 样 ， 但 由 于 基 类 不 同 而 导致 在 跨越 应 用 程序 域 的 边界 时 表现 出 的 行 
为 将 大 不 相同 。 

先 考虑 A 的 情况 。 由 于 A 继承 和 白 MarshalByRefObject， 那 么 a 实际 上 
只 是 在 默认 的 域 中 的 一 个 代理 实例 (Proxy) ， 它 指 同 位 于 NewDomain 
域 中 的 A 的 一 个 实例 。 当 调用 a 的 方法 SetNumber 时 ， 是 在 NewDomain 域 
中 调用 该 方法 ， 它 将 修改 NewDomain 域 中 静态 变量 A.Number 的 值 并 设 
为 20。 由 于 静态 变量 在 每 个 应 用 程序 域 中 都 有 一 份 独立 的 拷贝 ， 修 改 
NewDomain 域 中 的 静态 变量 A.Number 对 默认 域 中 的 静态 变量 A.Number 
没有 任何 影响 。 由 于 Console.WriteLine 是 在 默认 的 应 用 程序 域 中 输出 
A.Number， 因 此 输出 仍然 是 10。 

接着 讨论 B。 由 于 B 只 是 从 Object 继承 而 来 的 类 型 ， 它 的 实例 罕 越 应 
用 程序 域 的 边界 时 ， 将 会 完整 地 复制 实例 。 因 此 在 上 述 代 码 中 ， 我 们 尽 
管 试 图 在 NewDomain 域 中 生成 B 的 实例 ， 但 会 把 实例 b 复 制 到 默认 的 应 
用 程序 域 。 此 时 调用 方法 b.SetNumber 也 是 在 缺 省 的 应 用 程序 域 上 进 
行 ， 它 将 修改 默认 的 域 上 的 A.Number 并 设 为 20。 再 在 默认 的 域 上 调用 
Console.WriteLine 时 ， 它 将 输出 20。 

下 面 推荐 两 本 C# 相 关 的 书籍 ， 以 方便 大 家 应 对 C# 面 试 并 学 习 好 
CH. 

e (Professional C#》。 这 本 书 最 大 的 特点 是 在 附录 中 有 几 章 专 
门 写 给 已 经 有 其 他 语言 (如 VB、C++ 和 Java) 经 验 的 程序 员 ， 它 详细 讲 
ee 看 了 这 几 章 之 后 就 不 会 把 C# 和 之 前 掌握 的 
if BARA 

e Jeffrey Richter) (CLR Via C#) 。 该 书 不 仅 深 入 地 介绍 了 C# 
语言 ， 同 时 对 CLR 及 .NET 做 了 全 面 的 剖析 。 如 果 能 够 读 懂 这 本 书 ， 那 么 
我 们 就 能 深入 理解 装 箱 番 箱 、 垃 圾 回收 、 反 射 等 概念 ， 知 其 然 的 同时 也 
能 知 其 所 以 然 ， 通 过 C# 相 关 的 面试 自然 也 就 不 难 了 。 


面试 题 2， 实现 Singleton 模 式 


题目 : 设计 一 个 类 ， 我 们 只 能 生成 该 类 的 一 个 实例 。 

只 能 生成 一 个 实例 的 类 是 实现 了 Singleton A 模式 的 类 型 。 由 
于 设计 模式 在 面 癌 对 象 程 序 设计 中 起 着 举足轻重 的 作用 ， 在 面试 过 程 中 
很 多 公司 都 喜欢 问 一 些 与 设计 模式 相关 的 问题 。 在 向 用 的 模式 中 ， 









































Singleton 是 唯一 一 个 能 够 用 短 短 几 十 行 代码 完整 实现 的 模式 。 因 此 ， 写 
一 个 Singleton 的 类 型 是 一 个 很 常见 的 面试 题 。 


不 好 的 解法 一 ， 只 适用 于 单线 程 环境 


由 于 要 求 只 能 生成 一 个 实例 ， 因 此 我 们 必须 把 构造 函数 设 为 私有 函 
数 以 禁止 他 人 创建 实例 。 我 们 可 以 定义 一 个 静态 的 实例 ， 在 需要 的 时 候 
创建 该 实例 。 下 面 定 义 类 型 Singleton1 就 是 基于 这 个 思路 的 实现 : 
public sealed class Singletonl 
{ 





private static Singletonl instance = null; 
public static Singletonl Instance 
{ 
get 
{ 
if (instance == null) 
instance = new Singletonl(); 


return instance; 


} 
} 


上 述 代码 在 Singleton 的 静态 属性 Instance 中 ， 只 有 在 instance 为 null 的 
时 候 才 创建 一 个 实例 以 避免 重复 创建 。 同 时 我 们 把 构造 函数 定义 为 私有 
函数 ， 这 样 就 能 确保 只 创建 一 个 实例 。 


不 好 的 解法 二 : 虽然 在 多 线程 环境 中 能 工作 但 效率 不 局 


解法 一 中 的 代码 在 单线 程 的 时 候 工 作 正 常 ， 但 在 多 线程 的 情况 下 就 
有 问题 了 。 设 想 如 果 两 个 线程 同时 运行 到 判断 instance 是 个 为 null 的 证 语 
句 ， 并 且 instance 的 确 没有 创建 时 ， 那 么 两 个 线程 都 会 创建 一 个 实例 ， 
此 时 类 型 Singleton1 就 不 再 满足 单 例 模式 的 要 求 了 。 为 了 保证 在 多 线程 
环境 下 我 们 还 是 只 能 得 到 类 型 的 一 个 实例 ， 需 要 加 上 一 个 同步 锁 。 把 
Singleton1 稍 做 修改 得 到 了 如 下 代码 : 








public sealed class Singleton2 
{ 

private Singleton2 () 

1 


} 
private static readonly object syncObj = new object (); 


private static Singleton2 instance = null; 
public static Singleton2 Instance 
{ 

get 

{ 


lock (syncObj) 
{ 


if (instance == null) 
instance = new Singleton2(); 


} 


return instance; 


我 们 还 是 假设 有 两 个 线程 同时 想 创建 一 个 实例 。 由 于 在 一 个 时 刻 只 
有 一 个 线程 能 得 到 同步 锁 ， 当 第 一 个 线程 加 上 锁 时 ， 第 二 个 线程 只 能 等 
待 。 当 第 一 个 线程 发 现实 例 还 没有 创建 时 ， 它 创建 出 一 个 实例 。 接 着 第 
一 个 线程 释放 同步 锁 ， 此 时 第 二 个 线程 可 以 加 上 同步 锁 ， 并 运行 接 下 来 
的 代码 。 这 个 时 候 由 于 实例 已 经 被 第 一 个 线程 创建 出 来 了 ， 第 二 个 线程 
WR i 
一 个 实例 。 

但 是 类 型 Singleton2 还 不 是 很 完美 。 我 们 每 次 通过 属性 Instance 得 到 
Singleton2 的 实例 ， 都 会 试图 加 上 一 个 同步 锁 ， 而 加 锁 是 一 个 非常 耗 时 
的 操作 ， 在 没有 必要 的 时 候 我 们 应 该 尽量 避免 。 


可 行 的 解法 : 加 同步 锁 前 后 两 次 判断 实例 是 否 已 存在 
我 们 只 是 在 实例 还 没有 创建 之 前 需要 加 锁 操作 ， 以 保证 只 有 一 个 线 


程 创建 出 实例 。 而 当 实 例 已 经 创建 之 后 ， 我 们 已 经 不 需要 再 做 加 锁 操 作 
了 。 于 是 我 们 可 以 把 解法 二 中 的 代码 再 做 进一步 的 改进 : 








public sealed class Singleton3 
{ 


private Singleton3() 
} 


private static object syncobj = new object () ; 


private static Singleton3 instance = null; 
public static Singleton3 Instance 
i 


get 
{ 
if (instance == null) 
{ 
lock (syncObj) 
{ 
if (instance == null) 
instance = new Singleton3(); 
} 
} 


return instance; 


} 

Singleton3 中 只 有 当 instance 为 null 即 没有 创建 时 ， 需 要 加 锁 操 作 。 当 
instance 已 经 创建 出 来 之 后 ， 则 无 须 加 锁 。 因 为 只 在 第 一 次 的 时 候 
instance 为 null， 因 此 只 在 第 一 次 试图 创建 实例 的 时 候 需 要 加 锁 。 这 样 
Singleton3 的 时 间 效 率 比 Singleton2 要 好 很 多 。 

Singleton3 用 加 锁 机 制 来 确保 在 多 线程 环境 下 只 创建 一 个 实例 ， 并 
且 用 两 个 过 判断 来 提高 效率 。 这 样 的 代码 实现 起 来 比较 复杂 ， 容 易 出 
错 ， 我 们 还 有 更 加 优秀 的 解法 。 


强烈 推荐 的 解法 一 : 利用 静态 构造 函数 


C# 的 语法 中 有 一 个 函数 能 够 确保 只 调用 一 次 ， 那 就 是 静态 构造 函 
数 ， 我 们 可 以 利用 C# 这 个 特性 实现 单 例 模式 如 下 : 





{ 
Singleton5 () 
{ 
} 
public static Singleton5 Instance 
{ 
get 
{ 
return Nested.instance 
} 
} 
class Nested 
{ 
static Nested() 
{ 
} 
internal static readonly Singleton5 instance = new Singleton5(); 
} 
} 


Singleton4 的 实现 代码 非常 简 活 。 我 们 在 初始 化 静态 变量 instance 的 
时 候 创 建 一 个 实例 。 由 于 C# 是 在 调用 静态 构造 函数 时 初始 化 静态 变 
量 ，.NET 运 行 时 能 够 确保 只 调用 一 次 静态 构造 函数 ， 这 样 我 们 就 能 够 保 
证 只 初始 化 一 次 instance。 

C# 中 调用 静态 构造 函数 的 时 机 不 是 由 程序 员 擎 控 的 ， 而 是 当 .NET 
运行 时 发 现 第 一 次 使 用 一 个 类 型 的 时 候 自 动 调用 该 类 型 的 静态 构造 函 
数 。 因 此 在 Singleton4 中 ， 实 例 instance 并 不 是 第 一 次 调用 属性 
Singleton4.Instance 的 时 候 创 建 ， 而 是 在 第 一 次 用 a 到 Singleton4 的 时 候 就 
会 被 创建 。 假 设 我 们 在 Singleton4 中 添加 一 个 静态 方法 ， 调 用 该 静态 函 
数 是 不 需要 创建 一 个 实例 的 ， 但 如 果 按 照 Singleton4 的 方式 实现 单 例 模 
式 ， 则 仍然 会 过 早 地 创建 实例 ， 从 而 降低 内 存 的 使 用 效率 。 


强烈 推荐 的 解法 二 : 实现 按 需 创建 实例 


最 后 的 一 个 实现 Singleton5 则 很 好 地 解决 了 Singleton4 中 的 实例 创建 
时 机 过 早 的 问题 : 











public sealed class Singleton5 
{ 


Singleton5 () 


public static Singleton5 Instance 


internal static readonly Singleton5 instance = new Singleton5(); 
} 
} 


在 上 述 Singleton5 的 代码 中 ， 我 们 在 内 部 定义 了 一 个 私有 类 型 
Nested。 当 第 一 次 用 到 这 个 答 套 类 型 的 时 候 ， 会 调用 静态 构造 函数 创建 
Singleton5 的 实例 instance。 类 型 Nested 只 在 属性 Singleton5.Instance 中 被 
用 到 ， 由 于 其 私有 属性 他 人 无 法 使 用 Nested 类 型 。 因 此 当 我 们 第 一 次 试 
图 通过 属性 Singleton5.Instance 得 到 Singleton5 的 实例 时 ， 会 自动 调用 
Nested 的 静态 构造 函数 创建 实例 instance。 如 采 我 们 不 调用 属性 
Singleton5.Instance， 那 么 束 不 会 触发 .NET 运 行 时 调用 Nested， 也 不 会 创 
建 实例 ， 这 样 束 真正 做 到 了 按 需 创建 。 


解法 比较 


在 前 面 的 5 种 实现 单 例 模式 的 方法 中 ， 第 一 种 方法 在 多 线程 环境 中 
不 能 正常 工作 ， 第 二 种 模式 虽然 能 在 多 线程 环境 中 正常 工作 但 时 间 效 率 
很 低 ， 都 不 是 面试 官 期 待 的 解法 。 在 第 三 种 方法 中 我 们 通过 两 次 判断 一 
次 加 锁 确 保 在 多 线程 环境 能 高 效率 地 工作 。 第 四 种 方法 利用 C# 的 静态 构 
造 函 数 的 特性 ， 确 保 只 创建 一 个 实例 。 第 五 种 方法 利用 私有 髓 套 类 型 的 
特性 ， 做 到 只 在 真正 需要 的 时 候 才 会 创建 实例 ， 提 高 空间 使 用 效率 。 如 
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源 代码 : 
本 题 完 整 的 源 代码 详 见 02_Singleton 项 目 。 
本 题 考点 : 


e 考查 对 单 例 (Singleton) 模式 的 理解 。 
o 考查 对 C# 的 基础 语法 的 理解 ， 如 静态 构造 函数 等 。 
© 考查 对 多 线程 编程 的 理解 。 


本 题 扩展 : 


在 前 面 的 代码 中 ，5 种 单 例 模式 的 实现 把 类 型 标记 为 sealed， 表 示 它 
们 不 能 作为 其 他 类 型 的 基 类 。 现 在 我 们 要 求 定 义 一 个 表示 总 统 的 类 型 
President， 可 以 从 该 类 型 继承 出 FrenchPresident 和 AmericanPresident 等 类 
M, EMEA A BEET SRB. HAAS Bt SBIR ES 
型 ? 


2.3 数据 结构 


数据 结构 一 直 是 技术 面试 的 重点 ， 大 多 数 面试 题 都 是 围绕 着 数组 、 
字符 串 、 链 表 、 树 、 栈 及 队列 这 几 种 常见 的 数据 结构 展开 的 ， 因 此 每 一 
个 应 聘 者 都 要 熟练 掌握 这 几 种 数据 结构 。 

数组 和 字符 串 是 两 种 最 基本 的 数据 结构 ， 它 们 用 连续 内 存 分 别 存储 
数字 和 字符 。 链 表 和 树 是 面试 中 出 现 频率 最 高 的 数据 结构 。 由 于 操作 链 
表 和 树 需 要 操作 大 量 的 指针 ， 应 聘 者 在 解决 相关 问题 的 时 候 一 定 要 留意 
代码 的 鲁 棒 性 ， 否 则 容易 出 现 程序 骨 溃 的 问题 。 栈 是 一 个 与 递归 紧密 相 
天 的 数据 结构 ， 同 样 队 列 也 与 广度 优先 过 历 算法 紧密 相关 。 深 刻 理 解 这 
两 种 数据 结构 能 帮助 我 们 解决 很 多 算法 问题 。 


2.3.1 数组 


数组 可 以 说 是 最 简单 的 一 种 数据 结构 ， 它 占据 一 块 连续 的 内 存 并 按 
照 顺 序 存 储 数 据 。 创 建 数 组 时 ， 我 们 需要 首先 指定 数组 的 容量 大 小 ， 然 
后 根据 大 小 分 配 内 存 。 即 使 我 们 只 在 数组 中 存储 一 个 数字 ， 也 需要 为 所 
有 的 数据 预先 分 配 内 存 。 因 此 数组 的 空间 效率 不 是 很 好 ， 经 常会 有 空间 





























的 区 域 没 有 得 到 充分 利用 。 

由 于 数组 中 的 内 存 是 连续 的 ， 于 是 可 以 根据 下 标 在 O (1) 时间 读 / 
写 任何 元 素 ， 因 此 它 的 时 间 效 率 是 很 高 的 。 我 们 可 以 根据 数组 时 间 效 率 
高 的 优点 ， 用 数组 来 实现 简单 的 哈 希 表 : 把 数组 的 下 标 设 为 哈 希 表 的 键 
值 (Key) ， 而 把 数组 中 的 每 一 个 数字 设 为 哈 希 表 的 值 (Value) ， 这 样 
每 一 个 下 标 及 数组 中 该 下 标 对 应 的 数字 束 组 成 了 一 个 键 值 一 值 的 配对 。 
有 了 这 样 的 哈 希 表 ， 我 们 就 可 以 在 O (1) 实现 查找 ， 从 而 可 以 快速 高 
~、 问题 。 面 试题 35“ 第 一 个 只 出 现 一 次 的 字母 ?就 是 一 个 很 好 
例子。 

为 了 解决 数组 空间 效率 不 高 的 问题 ， 人 们 又 设计 实现 了 多 种 动态 数 
组 ， 比 如 C++ 的 STL 中 的 vector。 为 了 避免 浪费 ， 我 们 先 为 数组 开辟 较 小 
的 空间 ， 然 后 往 数 组 中 添加 数据 。 当 数据 的 数 日 超过 数组 的 容量 时 ， 我 
们 再 重新 分 配 一 块 更 大 的 空间 (STL 的 vector 每 次 扩充 容量 时 ， 新 的 容 
量 都 是 前 一 次 的 两 倍 ) ， 把 之 前 的 数据 复制 到 新 的 数组 中 ， 再 把 之 前 的 
内 存 释 放 ， 这 样 束 能 减少 内 存 的 浪费 。 但 我 们 也 注意 到 每 一 次 扩充 数组 
容量 时 都 有 大 量 的 额外 操作 ， 这 对 时 间 性 能 有 负面 影响 ， 因 此 使 用 动态 
数组 时 要 尽量 减少 改变 数组 容量 大 小 的 次 数 。 

在 C/C++ 中 ， 数 组 和 指针 是 相互 关联 义 有 区 别 的 两 个 概念 。 当 我 们 
声明 一 个 数组 时 ， 其 数组 的 名 字 也 是 一 个 指针 ， 访 指针 指 问 数 组 的 第 一 
个 元 素 。 我 们 可 以 用 一 个 指针 来 访问 数组 。 但 值得 注意 的 是 ，C/C++ 没 
有 记录 数组 的 大 小 ， 因 此 用 指针 访问 数组 中 的 元 素 时 ， 程 序 员 要 确保 没 
有 超出 数组 的 边界 。 下 面 通过 一 个 例子 来 了 解数 组 和 指针 的 区 别 。 运 行 
下 面 的 代码 ， 请 问 输出 是 什么 ? 



























































int GetSize(int Qata[]) 


return sizeof (data); 


int _tmain(int argc, _TCHAR* argv[]) 


int datal[] 


an t1; 2s 3y J SF 
int sizel = siz 


zeof (datal); 


nt* data2 = datal; 

int size2 = sizeof (data2); 

int size3 = GetSize(datal); 

printf ("t$d, td, td", sizel, size2, size3); 


} 

答案 是 输出 “20,4,4”。datal 是 一 个 数组 ，sizeof (datal) 是 求 数组 的 
大 小 。 这 个 数组 包含 5 个 整数 ， 每 个 整数 占 4 字 节 ， 因 此 总 共 是 20 字 他。 
data2 声 明 为 指针 ， 尽 管 它 指向 了 数组 datal 的 第 一 个 数字 ， 但 它 的 本 质 
仍然 是 一 个 指针 。 在 32 位 系统 上 ， 对 任意 指针 求 sizeof， 得 到 的 结果 都 
是 4。 在 C/C++ 中 ， 当 数组 作为 函数 的 参数 进行 传递 时 ， 数 组 就 自动 退 
化 为 同类 型 的 指针 。 因 此 尽管 函数 GetSize 的 参数 data 被 声明 为 数组 ， 但 
它 会 退化 为 指针 ，size3 的 结果 仍然 是 4。 


面试 题 3: 二 维 数组 中 的 得 找 


题目 : 在 一 个 二 维 数组 中 ， 每 一 行 都 按照 从 左 到 右 递增 的 
顺序 排序 ， 每 一 列 都 按照 从 上 到 下 递增 的 顺序 排序 。 请 完成 一 
个 函数 ， 输 入 这 样 的 一 个 二 维 数组 和 一 个 整数 ， 判 断 数组 中 是 
个 含有 该 整数 。 

例如 下 面 的 二 维 数组 就 是 每 行 、 每 列 都 递增 排序 。 如 果 在 这 个 数组 
中 查找 数字 7， 则 返回 true; 如 果 碍 找 数字 5， 由 于 数组 不 含有 该 数字 ， 
则 返回 false。 


























1 2 8 9 

2 4 9 12 
4 F 10 13 
6 8 a 35 


在 分 析 这 个 问题 的 时 候 ， 很 多 应 聘 者 都 会 把 二 维 数组 男 成 矩形 ， 然 
后 从 数组 中 选取 一 个 数字 ， 分 3 种 情况 来 分 析 碍 找 的 过 程 。 当 数组 中 选 
取 的 数字 刚好 和 要 碍 找 的 数字 相等 时 ， 就 结束 查找 过 程 。 如 果 选 取 的 数 
字 小 于 要 查找 的 数字 ， 那 么 根据 数组 排序 的 规则 ， 要 查找 的 数字 应 该 在 
当前 选取 的 位 置 的 右边 或 者 下 边 ( 如 图 2.1 ad Aa o AE, MRE 
取 的 数字 大 于 要 但 找 的 数字 ， 那 么 要 查找 的 数字 应 该 在 当前 选取 的 位 置 
的 上 边 或 者 左边 如 图 2.1 b MaR) o 
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a) 数组 中 的 数字 
要 查找 的 数字 


Cb) 数组 中 的 效 子 大 于 
要 查找 的 数字 





图 2.1 二 维 数 组 中 的 查找 


注 : 在 数组 中 间 选 择 一 个 数 〈 深 色 方 格 ) ， 根 据 它 的 大 小 判断 要 查找 的 数字 可 能 出 现 的 
区 域 (阴影 部 分 〉。 


在 上 面 的 分 析 中 ， 由 于 要 查找 的 数字 相对 于 当前 选取 的 位 置 有 可 能 
在 两 个 区 域 中 出 现 ， 而 且 这 两 个 区 域 还 有 重 妆 ， 这 问题 看 起 来 就 复杂 
了 ， 于 是 很 多 人 就 卡 在 这 里 束手无策 了 。 

当 我 们 需要 解决 一 个 复杂 的 问题 时 ， 一 个 很 有 效 的 办 法 就 是 从 一 个 
有 具体 的 问题 入 手 ， 通 过 分 析 简 单 具 体 的 例子 ， 试 图 寻找 普遍 的 规律 。 针 
对 这 个 问题 ， 我 们 不 妨 也 从 一 个 具体 的 例子 入 手 。 下 面 我 们 以 在 题目 中 
给 出 的 数组 中 查找 数字 7 为 例 来 一 步 步 分 析 查 找 的 过 程 。 

前 面 我 们 之 所 以 遇 到 难题 ， 是 因为 我 们 在 二 维 数组 的 中 间 选 取 一 个 
数字 来 和 要 查找 的 数字 做 比较 ， 这 样 导 致 下 一 次 要 查找 的 是 两 个 相互 重 
合 的 区 域 。 如 果 我 们 从 数组 的 一 个 角 上 选取 数字 来 和 要 查找 的 数字 做 比 
较 ， 情 况 会 不 会 变 简 单 呢 ? 

首先 我 们 选取 数组 右上 角 的 数字 9。 由 于 9 大 于 7， 并 且 9 还 是 第 4 列 
的 第 一 个 (也 是 最 小 的 ) 数字 ， 因 此 7 不 可 能 出 现在 数字 9 所 在 的 列 。 于 
是 我 们 把 这 一 列 从 需要 考虑 的 区 域内 剔除 ， 之 后 只 需要 分 析 剩 下 的 3 列 
(如 图 2.2 (a) 所 示 ) 。 在 剩 下 的 矩阵 中 ， 位 于 右上 角 的 数字 是 8。 同 样 
8 大 于 7， 因 此 8 所 在 的 列 我 们 也 可 以 剔除 。 接 下 来 我 们 只 要 分 析 剩 下 的 
两 列 即 可 〈 如 图 2.2 (b) 所 示 ) 。 

在 由 剩余 的 两 列 组 成 的 数组 中 ， 数 字 2 位 于 数组 的 右上 角 。2 小 于 
7， 那 么 要 查找 的 7 可 能 在 2 的 右边 ， 也 有 可 能 在 2 的 下 边 。 在 前 面 的 步 又 
中 ， 我 们 已 经 发 现 2 右 边 的 列 都 已 经 被 剔除 了 ， 也 就 是 说 7 不 可 能 出 现在 
2 的 右边 ， 因 此 7 只 有 可 能 出 现在 2 的 下 边 。 于 是 我 们 把 数字 2 所 在 的 行 也 
剔除 ， 只 分 析 剩 下 的 三 行 两 列 数字 〈 如 图 2.2 Cc) 所 示 ) 。 在 剩 下 的 数 
字 中 ， 数 字 4 位 于 右上 角 ， 和 前 面 一 样 ， 我 们 把 数字 4 所 在 的 行 也 删除 ， 
最 后 剩 下 两 行 两 列 数字 〈 如 图 2.2 (d) 所 示 ) 。 
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2 国 9 12 
4 7 10 13 
6 8 11 15 
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次 只 项 要 在 9 的 诺 边 
区 域 查找 
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Ge} 2eh7,.. Fo 
次 只 需要 在 2 的 下 边 
区 域 查找 
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Cd? 42h Fs EO 
次 只 需要 在 4 的 下 边 
区 域 查 找 
图 2.2 在 二 维 数组 中 查找 7 的 步 又 
VE: 矩阵 中 加 阴影 背景 的 区 域 是 下 一 步 查找 的 范围。 


在 剩 下 的 两 行 两 列 4 个 数字 中 ， 位 于 右上 角 的 刚好 就 是 我 们 要 碍 找 
的 数字 7， 于 是 查找 过 程 就 可 以 结束 了 。 





总 结 上 述 查 找 的 过 程 ， 我 们 发 现 如 下 规律 首先 选取 数组 中 右上 和 角 
的 数字 。 如 果 该 数字 等 于 要 查找 的 数字 ， 查 找 过 程 结 束 ; 如 果 该 数字 大 
于 要 查找 的 数字 ， 吻 除 这 个 数字 所 在 的 列 ， 如 果 该 数字 小 于 要 查找 的 数 
字 ， 剔 除 这 个 数字 所 在 的 行 。 也 束 是 说 如 果 要 查找 的 数字 不 在 数组 的 右 
上 角 ， 则 每 一 次 都 在 数组 的 查找 范围 中 剔除 一 行 或 者 一 列 ， 这 样 每 一 步 
都 可 以 缩小 查找 的 范围 ， 直 到 找到 要 碍 找 的 数字 ， 或 者 查找 范围 为 空 

把 整个 查找 过 程 分 析 清 楚 之 后 ， 我 们 再 写 代码 就 不 是 一 件 很 难 的 事 
i.s eres 述 ee kas 











bool Find(int* matrix, int row int columns, int number} 
{ 
bool found = false; 
if (matrix l= NULL BE rows > 0 && columns > 0) 
{ 
int row = 0; 
int column = columns - 1; 
while(row < rows && column >=0) 
{ 
if(matrix[row * columns + column] == number) 


{ 
found = true; 
break; 





在 前 面 的 分 析 中 ， 我 们 每 一 次 都 是 选取 数组 碍 找 范 围 内 的 右上 角 数 
字 。 同 样 ， 我 们 也 可 以 选取 左下 角 的 数字 。 感 兴趣 的 读者 不 妨 自 己 分 析 
一 咎 每 次 部 选取 左下 角 的 查找 过 程 。 但 我 们 不 能 选择 左上 角 或 者 右 下 
角 。 以 左上 角 为 例 ， 最 初 数字 1 位 于 初始 数组 的 左上 角 ， 由 于 1 小 于 7， 
那么 7 应 该 位 于 1 的 右边 或 者 下 边 。 此 时 我 们 既 不 能 从 碍 找 范围 内 剔除 1 
所 在 的 行 ， 也 不 能 剔除 1 所 在 的 列 ， 这 样 我 们 束 无 法 综 小 奏 找 的 范围 。 


源 代码 : 
本 题 完整 的 源 代码 详 见 03_FindInPartiallySortedMatrix 项 目 。 





测试 用 例 : 


e 二 维 数 组 中 包含 查找 的 数字 (查找 的 数字 是 数组 中 的 最 大 值 和 
最 小 值 ， 碍 找 的 数字 介 于 数组 中 的 最 大 值 和 最 小 值 之 间 ) 。 

e 二 维 数组 中 没有 碍 找 的 数字 【〈 碍 找 的 数字 大 于 数组 中 的 最 大 
值 ， 碍 找 的 数字 小 于 数组 中 的 最 小 值 ， 碍 找 的 数字 在 数组 的 最 大 值 和 最 
小 值 之 间 但 数组 中 没有 这 个 数字 〉。 

e 特殊 输入 测试 (输入 空 指 针 )。 


本 题 考点 : 


o 考 俘 应 聘 者 对 二 维 数 组 的 理解 及 编程 能 力 。 二 维 数 组 在 内 存 中 
占据 连续 的 空间 。 在 内 存 中 从 上 到 下 存储 各 行 元 素 ， 在 同一 行 中 按照 从 
左 到 右 的 顺序 存储 。 因 此 我 们 可 以 根据 行 写 和 列 写 计算 出 相对 于 数组 首 
地 址 的 偏 移 量 ， 从 而 找到 对 应 的 元 系 。 

© 考查 应 聘 者 分 析 问 题 的 能 力 。 当 应 聘 者 发 现 问 题 比 较 复 杂 时 ， 
能 不 能 通过 具体 的 例子 找 出 其 中 的 规律 ， 是 能 否 解 决 这 个 问题 的 关键 所 
人 在。 这 个 题目 只 要 从 一 个 具体 的 二 维 数组 的 右上 和 角 开 始 分 析 ， 束 能 找到 
查找 的 规律 ， 从 而 找到 解决 问题 的 突破 口 。 


2.3.2 APR 


字符 串 是 由 若干 字符 组 成 的 序列 。 由 于 字符 串 在 编程 时 使 用 的 频率 
非常 高 ， 为 了 优化 ， 很 多 语言 都 对 字符 串 做 了 特殊 的 规定 。 下 面 分 别 讨 
论 C/C++ 和 C# 中 字符 串 的 特性 。 

C/C++ 中 每 个 字符 串 都 以 字符 \0' 作 为 结尾 ， 这 样 我 们 束 能 很 方便 地 
找到 字符 串 的 最 后 尾部 。 但 由 于 这 个 特点 ， 每 个 字符 串 中 都 有 一 个 额外 
字符 的 开销 ， 稍 不 留神 就 会 造成 字符 串 的 越界 。 比 如 下 面 的 代码 : 
char str[10]; 
strepy(str, 0123456789"); 

我 们 先 声明 一 个 长 度 为 10 的 字符 数组 ， 然 后 把 字符 
串 "0123456789" 复 制 到 数组 中 。"0123456789" 这 个 字符 串 看 起 来 只 有 10 
个 字符 ， 但 实际 上 它 的 末尾 还 有 一 个 \0' 字 符 ， 因 此 它 的 实际 长 度 为 11 个 
字 节 。 要 正确 地 复制 该 字符 串 ， 至 少 需要 一 个 长 度 为 11 个 字 节 的 数组 。 

为 了 节省 内 存 ，C/C++ 把 常量 字符 串 放 到 单独 的 一 个 内 存 区 域 。 当 
几 个 指针 赋值 给 相同 的 常量 字符 串 时 ， 它 们 实际 上 会 指向 相同 的 内 存 地 
址 。 但 用 常量 内 存 初始 化 数组 ， 情 况 却 有 所 不 同 。 下 面 通过 一 个 面试 题 
来 学 习 这 一 知识 点 。 运 行 下 面 的 代码 ， 得 到 的 结果 是 什么 ? 
































int tmain(int argc, _TCHAR* argv[]) 


{ 
char stri[] = “hello world"; 
char str2[] = “hello world"; 
char* str3 = "hello world" 
char* str4 = “hello rid" 
if (strl == str2) 
printf ("stri and str2 are same.\n"); 
print£("strl and str2 are not same.\n"); 
if(str3 == strå) 
printf ("str3 and str4 are same.\n"); 
printf ("str3 and str4 are not same.\n"); 
return 0; 
} 


str1 和 str2 是 两 个 字符 串 数 组 ， 我 们 会 为 它们 分 配 两 个 长 度 为 12 个 字 
节 的 空间 ， 并 把 "hello world" 的 内 容 分 别 复制 到 数组 中 去 。 这 是 两 个 初 
始 地 址 不 同 的 数组 ， 因 此 str1 和 str2 的 值 也 不 相同 ， 所 以 输出 的 第 一 行 
是 ”strl and str2 are not same”. 

str3 和 str4 是 两 个 指针 ， 我 们 无 须 为 它们 分 配 内 存 以 存储 字符 串 的 内 
容 ， 而 只 需要 把 它们 指向 "hello world” 在 内 存 中 的 地 址 就 可 以 了 。 由 
于 "hello ”world” 是 常量 字符 串 ， 它 在 内 存 中 只 有 一 个 拷贝 ， 因 此 str3 和 
str4 指 癌 的 是 同一 个 地 址 。 所 以 比较 str3 和 str4 的 值得 到 的 结果 是 相同 
的 ， 输 出 的 第 二 行 是 ”str3 and str4 are same”. 

在 C# 中 ， 封 装 字符 串 的 类 型 System.String 有 一 个 非常 特殊 的 性 质 : 
String 中 的 内 容 是 不 能 改变 的 。 一 旦 试图 改变 String 的 内 容 ， 束 会 产生 一 
个 新 的 实例 。 请 看 下 面 的 C# 代 码 : 

String str = "hello",; 
str.ToUpper (); 
str.Insert(0, " WORLD"); 

虽然 我 们 对 str 做 了 ToUpper 和 Insert 两 个 操作 ， 但 操作 的 结果 都 是 生 
成 一 个 新 的 String 实 例 并 在 返回 值 中 返回 ，str 本 身 的 内 容 都 不 会 发 生 改 
变 ， 因 此 最 终 str 的 值 仍然 是 "hello"。 由 此 可 见 ， 如 果 试 图 改变 String 的 
内 容 ， 改 变 之 后 的 值 只 可 以 通过 返回 值得 到 。 用 String 作 连续 多 次 修 
改 ， 每 一 次 修改 都 会 产生 一 个 临时 对 象 ， 这 样 开销 太 大 会 影响 效率 。 为 











此 C# 定 义 了 一 个 新 的 与 字符 串 相 关 的 类 型 StringBuilder， 它 能 容纳 修改 
后 的 结果 。 因 此 如 有 果 要 连续 多 次 修改 字符 串 内 容 ， 用 StringBuilder 征 更 
好 的 选择 。 

和 修改 String 内 容 类 似 ， 如 果 我 们 试图 把 一 个 常量 字符 串 赋 值 给 一 
个 String 实 例 ， 也 不 是 把 String 的 内 容 改 成 赋值 的 字符 串 ， 而 是 生成 一 个 
新 的 String 实 例 。 请 看 下 面 的 代码 : 


class Program 


{ 
internal static void ValueOrReference (Type type) 
{ 
String result = "The type " + type.Name; 
if (type.IsValueType) 
Console.WriteLine (result + " is a value type.") 
Console.WriteLine (result + " is a reference type."); 
} 


internal static void ModifyString (String text) 


{ 
text = "world"; 
} 
static void Main(string[] args) 
{ 
Q 1 gq text — " he " 
JValueOrReference (text.GetType ()) 
ModifyString (text); 
onsole.WriteLine (text) 
} 


在 上 面 的 代码 中 ， 我 们 先 判 断 String 是 值 类 型 还 是 引用 类 型 。 类 型 
String 的 定义 是 public sealed class String {...}。 既 然 是 class， 那 么 String 自 
然 就 是 引用 类 型 。 接 下 来 在 方法 ModifyString 里 ， 对 text 赋 值 一 个 新 的 字 
符 串 。 我 们 要 记得 text 的 内 容 是 不 能 被 修改 的 。 此 时 会 先生 成 一 个 新 的 
内 容 是 "world" 的 String 实 例 ， 然 后 把 text 指 向 这 个 新 的 实例 。 由 于 参数 
text 没 有 加 ref 或 者 out， 出 了 方法 ModifyString 之 后 ，text 还 是 指 同 原来 的 
字符 串 ， 因 此 输出 仍然 是 "hello"。 要 想 实现 出 了 函数 之 后 text 变 
成 "world" 的 效果 ， 我 们 必须 把 参数 text 标 记 ref 或 者 out。 


面试 题 4: 蔡 换 空格 


PAA: 请 实现 一 个 函数 ， 把 字符 串 中 的 每 个 空格 蔡 换 
成 "%20"。 例 如 输入 “We are happy.”， 则 输 
出 “We%20are%20happy.”。 

在 网 络 编程 中 ， 如 果 URL 参 数 中 含有 特殊 字符 ， 如 空格 、## 等 ， 可 
能 导致 服务 器 端 无 法 获得 正确 的 参数 值 。 我 们 需要 将 这 些 特殊 符号 转换 
成 服务 器 可 以 识别 的 字符 。 转 换 的 规则 是 在 '"%' 后 面 跟 上 ASCII 码 的 两 位 
十 六 进 制 的 表示 。 比 如 空格 的 ASCII 码 是 32， 即 十 六 进 制 的 0x20， 因 此 
空格 被 替换 成 "%20"。 再 比如 党 的 ASCII 码 为 35， 即 十 六 进 制 的 0x23， 它 
在 UREL 中 被 蔡 换 为 "9%23"。 

看 到 这 个 题目 ， 我 们 首先 应 该 想到 的 是 原来 一 个 空格 字符 ， 蔡 换 之 
后 变 成 '%'、'2' 和 '0' 这 3 个 字符 ， 因 此 字符 串 会 变 长 。 如 果 是 在 原来 的 字 
符 串 上 做 蔡 换 ， 那 么 就 有 可 能 上 履 盖 修改 在 该 字符 串 后 面 的 内 存 。 如 果 是 
创建 新 的 字符 串 并 在 新 的 字符 串 上 做 替换 ， 那 么 我 们 可 以 自己 分 配 足 够 
多 的 内 存 。 由 于 有 两 种 不 同 的 解决 方案 ， 我 们 应 该 同 面 试 官 问 清楚 ， 让 
他 明确 告诉 我 们 他 的 需求 。 假 设 面 试 官 让 我 们 在 原来 的 字符 串 上 做 蔡 
换 ， 并 且 保 证 输入 的 字符 串 后 面 有 足够 多 的 空余 内 存 。 

时 间 复 杂 上 度 为 O0 Cm) 的 解法 ， 不 足以 拿 到 Offer 

现在 我 们 考虑 怎么 做 奉 换 操 作 。 最 直观 的 做 法 是 从 头 到 尾 扫 拉 字符 
串 ， 每 一 次 人 页 到 空格 字符 的 时 候 做 蔡 换 。 由 于 是 把 1 个 字符 蔡 换 成 3 个 字 
符 ， 我 们 必须 要 把 空格 后 面 所 有 的 字符 都 后 移 两 个 字 节 ， 人 否则 就 有 两 个 
FIFE tt T o 

举 个 例子 ， 我 们 从 头 到 尾 把 "We are _ happy." 中 的 每 一 个 空格 替换 
成 "%20"。 为 了 形象 起 见 ， 我 们 可 以 用 一 个 表格 来 表示 字符 串 ， 表 格 中 
的 每 个 格子 表示 一 个 字符 《〈 如 图 2.3 Ca) 所 示 ) 。 
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图 2.3 ”从 前 往 后 把 字符 串 中 的 空格 蔡 换 成 '%20' 的 过 程 
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YE: (a) 字符 串 "We are happy."。 b) 把 字符 串 中 的 第 一 个 空格 替换 成 %20'。 灰 色 背 景 
表示 需要 移动 的 字符 。 (c) 把 字符 串 中 的 第 二 个 空格 替换 成 %20'。 浅 灰色 背景 表示 需要 移动 
一 次 的 字符 ， 深 灰色 背景 表示 需要 移动 两 次 的 字符 。 

我 们 蔡 换 第 一 个 空格 ， 这 个 字符 串 变 成 图 2.3 b 中 的 内 容 ， 表 格 
中 灰色 背景 的 格子 表示 需要 做 移动 的 区 域 。 接 着 我 们 替换 第 二 个 空格 ， 
替换 之 后 的 内 容 如 图 2.3 Cc) 所 示 。 同 时 ， 我 们 注意 到 用 深 灰 色 背 景 标 
注 的 "happy" 部 分 被 移动 了 两 次 。 

假设 字符 串 的 长 度 是 n。 对 每 个 空格 字符 ， 需 要 移动 后 面 O0 (n) 个 
字符 ， 因 此 对 侣 有 O (n) 个 空格 字符 的 字符 串 而 言 总 的 时 间 效 率 是 
O m). 

当 我 们 把 这 种 思路 前 述 给 面试 官 后 ， 他 不 会 就 此 满意 ， 他 将 让 我 们 
寻找 更 快 的 方法 。 在 前 面 的 分 析 中 ， 我 们 发 现 数组 中 很 多 字符 都 移动 了 
很 多 次 ， 能 不 能 减少 移动 次 数 呢 ? 答案 是 肯定 的 。 我 们 换 一 种 思路 ， 把 
从 前 同 后 蔡 换 改 成 从 后 同 前 蔡 换 。 


时 间 复 杂 度 为 O Cn) 的 解法 ， 搞 定 Offer 就 徘 它 了 


我 们 可 以 先 过 历 一 次 字符 串 ， 这 样 就 能 统计 出 字符 串 中 空格 的 总 
数 ， 并 可 以 由 此 计算 出 替换 之 后 的 字符 串 的 总 长 度 。 每 蔡 换 一 个 空格 ， 
长 度 增 加 2， 因 此 蔡 换 以 后 字符 串 的 长 度 等 于 原来 的 长 度 加 上 2 乘 以 空格 
数目 。 我 们 还 是 以 前 面 的 字符 串 "We are ”happy." 为 例 ，"We are 
happy." 这 个 字符 串 的 长 度 是 14 (包括 结尾 符 写 \0') ， 里 面 有 两 个 空 
格 ， 因 此 蔡 换 之 后 字符 串 的 长 度 是 18。 

我 们 从 字符 串 的 后 面 开 始 复制 和 蔡 换 。 首 先 准备 两 个 指针 ，P1 和 
P2。P1 指 向 原始 字符 串 的 末尾 ， 而 P2 指 向 替换 之 后 的 字符 串 的 末尾 〈 如 
图 2.4 Ca) 所 示 ) 。 接 下 来 我 们 同 前 移动 指针 P1， 逐 个 把 它 指 问 的 字符 
复制 到 P2 指 向 的 位 置 ， 直 到 碰 到 第 一 个 空格 为 止 。 此 时 字符 串 包含 如 图 
2.4 (b) 所 示 ， 灰 色 背 景 的 区 域 是 做 了 字符 拷贝 〈 移 动 ) 的 区 域 。 碰 到 
第 一 个 空格 之 后 ， 把 P1 同 前 移动 1 格 ， 在 P2 之 前 插入 字符 串 "%20"。 由 
于 "%20" 的 长 度 为 93， 同时 也 要 把 P2 回 前 移动 3 格 如 图 2.4 Cc) 所 示 。 

我 们 接着 向 前 复制 ， 直 到 们 到 第 二 个 空格 (如 图 2.4 Cd) 所 示 ) 。 
和 上 一 次 一 样 ， 我 们 再 把 P1 回 前 移动 1 格 ， 并 把 P2 疝 前 移动 3 格 插 
入 "%20"〈 如 图 2.4 Ce) Aras) 。 此 时 P1 和 P2 指 向 同一 位 置 ， 表 明 所 有 
空格 都 已 经 蔡 换 完毕 。 
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图 2.4 ”从 后 往 前 把 字符 串 中 的 空格 蔡 换 成 <%20” 的 过 程 
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: 图 中 带 有 阴影 的 区 域 表示 被 移动 的 字符 。(a)〉 把 第 一 个 指针 指向 字符 串 的 末尾 ， 把 
第 二 ae i 针 指 向 普 换 之 后 的 字符 囊 的 末尾 。 (b) 依次 复制 字符 串 的 内 容 ， 直 至 第 一 个 指针 磁 到 
第 二 个 空格 。 Cc) 把 第 一 个 空格 普 换 成 '%20'"， 把 第 一 个 指针 向 前 移动 1 格 ， 把 第 二 个 指针 向 前 
移动 3 格 。 Cd) 依次 向 前 复制 字符 串 中 的 字符 ， 直 至 碰 到 空格 。(e》 蔡 换 字符 串 中 的 倒数 第 二 
个 空格 ， 把 第 一 个 指针 向 前 移动 1 格 ， 把 第 二 个 指针 向 前 移动 3 格 。 


从 上 面 的 分 析 我 们 可 以 看 出 ， 所 有 的 字符 都 只 复制 (移动 ) 一 次 ， 
因此 这 个 算法 的 时 间 效 率 是 O(n) ， 比 第 一 个 思路 要 快 。 

在 面试 的 过 程 中 ， 我 们 也 可 以 和 前 面 的 分 析 一 样 画 一 两 个 示意 图 解 
释 自 己 的 思路 ， 这 样 既 能 帮助 我 们 理 清 思路 ， 也 能 使 我 们 和 面试 官 的 交 
流 变 得 更 加 高 效 。 在 面试 官 肯定 我 们 的 思路 之 后 ， 就 可 以 开始 写 代码 
了 了。 下面 是 参考 代码 : 









































/*length 为 字符 数组 string 的 总 容量 */ 
void ReplaceBlank(char string[], int length) 


{ 
if(string == NULL && length <= 0) 


return; 


/*originalLength AFA string 的 实际 长 度 */ 
int originalLength = 0; 
int numberOfBlank = 0; 
int i = 0; 
while(string[i] != '\0') 
{ 
++ originalLength; 


if(stringli] == e *) 
++ numberOfBlank; 


a ig 


} 


/*newLength Ade eBHAM'S20'ZSHKRE*/ 
int newLength = originalLength + numberOfBlank * 2; 
if(newLength > length) 

return; 


int indexOfOriginal = originalLength; 
int indexofNew = newLength; 
while (indexOfOriginal >= 0 && indexOfNew > indexOfOriginal) 
{ 
if (string[indexOfOriginal] == ' ') 
{ 
string[indexofNew --] 
string[indexofNew --] 
string[indexofNew --] 
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} 


else 


{ 


| 

| 
— 

Il 


string [indexofNew string [indexoOfOriginal]; 


} 


-- indexoOfOriginal; 


源 代码 : 
本 题 完 整 的 源 代 码 详 见 04_ReplaceBlank 项 目 。 
测试 用 例 : 


o 输入 的 字符 串 中 包含 空格 (空格 位 于 字符 串 的 最 前 面 ， 空 格 位 
于 字符 串 的 最 后 面 ， 空 格 位 于 字符 串 的 中 间 ， 字 符 串 中 有 连续 多 个 空 
格 ) 。 

。 输入 的 字符 吝 中 没有 空格 。 

。 特殊 输入 测试 (字符 囊 是 个 NULL 指 针 、 字 符 串 是 个 空 字符 
串 、 字 符 囊 只 有 一 个 空格 字符 、 字 符 串 中 只 有 连续 多 个 空格 》。 


本 题 考点 : 


© 考 香 对 字符 串 的 编程 能 力 。 

© 考查 分 析 时 间 效 紊 的 能 力 。 我 们 要 能 清晰 地 分 析出 两 种 不 同方 
法 的 时 间 效 京 各 是 多 少 。 

。 考 香 对 内 存 履 盖 是 否 有 高 度 的 警惕 。 在 分 析 得 知 字符 串 会 变 长 
0 
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们 能 迅速 想到 从 后 往 前 谷 换 的 方法 ， 这 和 是 解决 此 题 的 关键 。 


相关 题目 : 


有 两 个 排序 的 数组 A1 和 A2， 内 存在 Al 的 末尾 有 足够 多 的 空余 空间 
容纳 A2。 请 实现 一 个 函数 ， 把 A2 中 的 所 有 数字 插入 到 Al 中 并 且 所 有 的 
数字 是 排序 的 。 

和 前 面 的 例题 一 样 ， 很 多 人 首先 想到 的 办 法 是 在 A1 中 从 头 到 尾 复 
制 数 字 ， 但 这 样 就 会 出 现 多 次 复制 一 个 数字 的 情况 。 更 好 的 办 法 是 从 尾 
到 头 比较 A1 和 A2 中 的 数字 ， 并 把 较 大 的 数字 复制 到 Al 的 合适 位 置 。 

2 


合并 两 个 数组 〈 包 括 字符 串 ) 时 ， 如 果 从 前 往 后 复制 每 个 数字 《或 字符 ) 需要 重复 移动 
0 
效率 。 

































































2.3.3 ”链表 


链表 应 该 是 面试 时 被 提 及 最 频 党 的 数据 结构 。 链 表 的 结构 很 简单 ， 
它 由 指针 把 若干 个 结 点 连接 成 链 状 结构 。 链 表 的 创建 、 插 入 结 点 、 删 除 
结 点 等 操作 都 只 需要 20 行 左右 的 代码 惑 能 实现 ， 其 代码 量 比较 适合 面 
试 。 而 像 哈 希 表 、 有 癌 图 等 复杂 数据 结构 ， 实 现 它们 的 一 个 操作 需要 的 
代码 量 都 较 大 ， 很 难 在 几 十 分 钟 的 面试 中 完成 。 另 外 ， 由 于 链表 是 一 种 
动态 的 数据 结构 ， 其 操作 需要 对 指针 进行 操作 ， 因 此 应 聘 者 需要 有 较 好 
的 编程 功底 才能 写 出 完整 的 操作 链表 的 代码 。 而 且 链 表 这 种 数据 结构 很 
灵活 ， 面 试 官 可 以 用 链表 来 设计 具有 挑战 性 的 面试 题 。 基 于 上 述 几 个 原 
因 ， 很 多 面试 官 都 特别 青睐 链表 相关 的 题目 。 

我 们 说 链表 是 一 种 动态 数据 结构 ， 是 因为 在 创建 链表 时 ， 无 须知 道 
链表 的 长 度 。 当 插入 一 个 结 点 时 ， 我 们 只 需要 为 新 结 点 分 配 内 存 ， 然 后 
调整 指针 的 指向 来 确保 新 结 点 被 链接 到 链表 当中 。 内 存 分 配 不 是 在 创建 
链表 时 一 次 性 完成 ， 而 是 每 添加 一 个 结 点 分 配 一 次 内 存 。 由 于 没有 闲置 
的 内 存 ， 链 表 的 空间 效率 比 数组 高 。 如 果 单 癌 链 表 的 结 点 定义 如 下 : 
struct ListNode 
{ 

int m_enValue; 

ListNode* m_pNext; 

}; 

那么 往 该 链表 的 末尾 中 添加 一 个 结 点 的 C 语 言 代 码 如 下 : 

void AddToTail (ListNode** pHead, int value) 


{ 




















ListNode* pNew = new ListNode(); 
pNew->m_nValue = value; 
pNew->m_pNext = NULL; 


if (*pHead == NULL) 
{ 
*pHead = pNew; 


else 
{ 


ListNode* pNode = *pHead; 


while (pNode->m_pNext != NULL) 
pNode = pNode->m_pNext; 


pNode->m_pNext = pNew; 


在 上 面 的 代码 中 ， 我 们 要 特别 注意 函数 的 第 一 个 参数 pHead 是 一 个 
指 回 指 针 的 指针 。 当 我 们 往 一 个 空 链表 中 插入 一 个 结 点 时 ， 新 插入 的 结 
点 驶 是 链表 的 头 指 针 。 由 于 此 时 会 改动 头 指针 ， 因 此 必须 把 pHead 参 数 
设 为 指 加 指针 的 指针 ， 人 否则 出 了 这 个 函数 pHead 仍 然 是 一 个 空 指 针 。 

由 于 链表 中 的 内 存 不 是 一 次 性 分 配 的 ， 因而 我 们 无 法 保证 链表 的 内 
存 和 数组 一 样 是 连续 的 。 因 此 如 果 想 在 链表 中 找到 它 的 第 个 结 点 ， 我 
们 只 能 从 头 结 点 开始 ， 沿 着 指 问 下 一 个 结 扣 的 指针 亿 历 链表 ， 它 的 时 间 
效率 为 O(n) 。 而 在 数组 中 ， 我 们 可 以 根据 下 标 在 O (1) 时间 内 找到 
ee eee 第 一 个 含有 某 值 的 结 点 并 删除 该 结 点 


void RemoveNode (ListNode** pHead, int value) 

















{ 
if (pHead == NULL *pHead == NULL) 
return; 
ListNo e* Te es = NULL; 
Ept ee >m nValue == value) 


ListNode* pNode = *pHead; 

while (pNode->m_pNext != NULL 
&& pNode->m_ pNext->m nValue != value) 
pNode = pNode->m , pNext; 


if (pNode->m_pNext != NULL && pNode->m_pNext->m_nValue == value) 


oTOBeDeleted = pNode->m_pNext; 
pNode->m_pNext = pNode->m_pNext->m_pNext; 


除了 简单 的 单 癌 链表 经 利和 被 设计 为 面试 题 之 外 《面试 题 5 从 尾 到 头 


输出 链表 ” 面试 题 13“ 在 O (1) 时 间 删 除 链 表 结 点 ” 面试 题 15“ 链 表 中 
的 倒数 第 k 个 结 点 ” 面试 题 1“ 反 转 链表 ”、 面 试题 17“ 合 并 两 个 排序 的 
链表 ”、 面 试题 37“ 两 个 链表 的 第 一 个 公共 结 点 ”等 ) ， 链 表 的 其 他 形式 
同样 也 备 受 面 试 官 的 青睐 。 

e 把 链表 的 末尾 结 点 的 指针 指 癌 尖 结 点 ， 从 而 形成 一 个 环形 链表 
〈 面 试题 45“ 圆 固 中 最 后 剩 下 的 数字 ”) 。 

e 链表 中 的 络 点 中 除了 有 指 网 下 一 个 结 点 的 指针 ， 还 有 指 同 前 一 
人 
2) 








。 链表 中 的 结 点 中 除了 有 指向 下 一 个 结 点 的 指针 ， 还 有 指向 任意 
结 点 的 指针 ， 这 就 是 复杂 链表 (面试 题 26“ 复 杂 链 表 的 复制 >) 。 


面试 题 5: 从 尾 到 头 打印 链表 


题目 : 输入 一 个 链表 的 头绪 点， 从 尾 到 头 反 过 来 打印 出 每 
个 结 点 的 值 。 链 表 结 点 定义 如 下 : 
struct ListNode 
{ 


hi 

看 到 这 道 题 后 ， 很 多 人 的 第 一 反应 是 从 头 到 尾 输出 将 会 比较 简单 ， 
于 是 我 们 很 自然 地 想到 把 链表 中 链接 结 点 的 指针 反 转 过 来 ， 改 变 链表 的 
方向 ， 然 后 就 可 以 从 头 到 尾 输出 了 。 但 该 方法 会 改变 原来 链表 的 结构 。 
是 个 允许 在 打印 链表 的 时 候 修 改 链表 的 结构 ? 这 个 取决 于 面试 官 的 需 
求 ， 因 此 在 面试 的 时 候 我 们 要 询问 清楚 面试 官 的 要 求 。 

面试 小 提示 : 

在 面试 中 如 果 我 们 打算 修改 输入 的 数据 ， 最 好 先 问 面试 官 是 不 是 允许 做 修改 。 

通 音 打印 是 一 个 只 读 操作 ， 我 们 不 希望 打印 时 修改 内 容 。 假 设 面试 
官 也 要 求 这 个 题目 不 能 改变 链表 的 结构 。 

接 下 来 我 们 想到 解决 这 个 问题 肯定 要 授 历 链表 。 授 历 的 顺序 是 从 类 
到 尾 的 顺序 ， 可 输出 的 顺序 却 是 从 尾 到 头 。 也 就 是 说 第 一 个 所 历 到 的 结 
点 最 后 一 个 输出 ， 而 最 后 一 个 过 历 到 的 结 点 第 一 个 输出 。 这 了 就 是 典型 
的 “后 进 先 出 ”， 我 们 可 以 用 栈 实现 这 种 顺序 。 每 经 过 一 个 结 点 的 时 候 ， 
把 该 结 点 放 到 一 个 栈 中 。 当 过 历 完 整个 链表 后 ， 再 从 栈 顶 开始 逐个 输出 
结 点 的 值 ， 此 时 输出 的 结 点 的 顺序 已 经 反 转 过 来 了 。 这 种 思路 的 实现 代 









































码 如 下 : 


void PrintListReversingly_Iteratively (ListNode* pHead) 


{ 


std: :stack<ListNode*> nodes; 


ListNode* pNode = pHead; 
while (pNode != NULL) 
{ 
nodes.push (pNode) ; 
pNode = pNode->m_pNext; 


} 


while (!nodes.empty () ) 

{ 
pNode = nodes.top(); 
printf ("Sd\t", pNode—->m_nValue) ; 
nodes.pop(); 


} 


既然 想到 了 用 栈 来 实现 这 个 函数 ， 而 递归 在 本 质 上 就 是 一 个 栈 结 
构 ， 于 是 很 自然 地 又 想到 了 用 递归 来 实现 。 要 实现 反 过 来 输出 链表 ， 我 
们 每 访问 到 一 个 结 点 的 时 候 ， 先 递归 输出 它 后 面 的 结 点 ， 再 输出 该 结 点 
自身 ， 这 样 链 表 的 输出 结果 就 反 过 来 了 。 

基于 这 样 的 思路 ， 不 难 写 出 如 下 代码 : 
void PrintListReversingly Recursively(ListNode* pHead) 


{ 


n 


if(pHead != NULL) 
{ 
if (pHead->m pNext != NULL) 
{ 
PrintListReversingly Recursively (pHead->m_pNext) ; 


} 


printf ("%d\t", pHead->m_ nValue); 


} 


上 面 的 基于 递归 的 代码 看 起 来 很 简洁 ， 但 有 个 问题 : 当 链 表 非 常 长 
的 时 候 ， 束 会 导致 函数 调用 的 层级 很 深 MAA H Re T BEK A y H Betin 
出 。 显 式 用 栈 基 于 循环 实现 的 代码 的 鲁 棒 性 要 好 一 些 。 更 多 关于 循环 和 
递归 的 讨论 ， 详 见 本 书 的 2.4.2 节 。 


} 











测试 用 例 : 


e 功能 测试 〈 输 入 的 链表 有 多 个 结 点 ， 输 入 的 链表 只 有 一 个 结 
FA) 。 
e 特殊 输入 训 试 《输入 的 链表 头 结 点 指针 为 NULL ) 。 


本 题 考点 : 


@ 考 伍 对 单项 链表 的 理解 和 编程 能 
© 考查 对 循环 、 递 归 和 栈 3 个 相互 关联 的 概念 的 理解 。 


2.3.4 pif 


树 是 一 种 在 实际 编程 中 经 常 遇 到 的 数据 结构 。 它 的 逻辑 很 简单 : 除 
了 根 结 点 之 外 每 个 结 点 只 有 一 个 父 结 点 ， 根 结 点 没有 父 结 点 ; 除了 叶 结 
点 之 外 所 有 结 点 都 有 一 个 或 多 个 子 结 点 ， 叶 结 点 没有 子 结 点 。 父 结 点 和 
子 结 点 之 间 用 指针 链接 。 由 于 树 的 操作 会 涉及 大 量 的 指针 ， 因 此 与 树 有 
关 的 面试 题 都 不 太 容易 。 当 面试 官 想 考 查 应 聘 者 在 有 复杂 指针 操作 的 情 
况 下 写 代 码 的 能 力 ， 他 往往 会 想到 用 与 树 有 关 的 面试 题 。 

面试 的 时 候 提 到 的 树 ， 大 部 分 都 是 二 叉 树 。 所 谓 二 又 树 是 树 的 一 种 
特殊 结构 ， 在 二 叉 树 中 每 个 结 点 最 多 只 能 有 两 个 子 结 点 。 在 二 又 树 中 最 
重要 的 操作 莫 过 于 遍历 ， 即 按照 某 一 顺序 访问 树 中 的 所 有 结 点 。 通 常 树 
有 如 下 几 种 遍历 方式 : 

e HRA: 先 访问 根 结 点 ， 再 访问 左 子 结 点 ， 最 后 访问 右 子 结 
点 。 图 2.5 中 的 三 又 树 的 前 序 遍 历 的 顺序 是 10、6、4、8、14、12、16。 

e PFH: 先 访 问 左 子 结 点 ， 再 访问 根 结 点 ， 最 后 访问 右 子 结 
点 。 图 2.5 中 的 二 叉 树 的 中 序 遍 历 的 顺序 是 4、6、8、10、12、14、16。 

e ria: 先 访问 左 子 结 点 ， 再 访问 右 子 结 点 ， 最 后 访问 根 结 
点 。 图 2.5 中 的 三 叉 树 的 后 序 遍 历 的 顺序 是 4、8、6、12、16、14、10。 




















图 2.5 ”一 个 二 又 树 的 例子 


这 3 种 遍历 都 有 递归 和 循环 两 种 不 同 的 实现 方法 ， 每 一 种 遍历 的 递 
归 实 现 都 比 循环 实现 要 简捷 很 多 。 很 多 面试 官 喜欢 直接 或 间接 考查 遍历 
〈 面 试题 39“ 二 又 树 的 深度 ” 面试 题 18“ 树 的 子 结构 ” 面试 题 25“ 二 又 
树 中 和 为 某 一 值 的 路 径 >) 的 有 具体 代码 实现 ， 面 试题 6“ 重 建 二 又 树 ” 面 
试题 24“ 二 又 树 的 后 序 遍 历 序列 ”也 是 考查 对 遍历 特点 的 理解 ， 因 此 应 聘 
者 应 该 对 这 3 种 遍历 的 6 种 实现 方法 都 了 如 指 掌 。 

e 宽度 优先 遍历 : 先 访问 树 的 第 一 层 结 点 ， 再 访问 树 的 第 二 层 结 
点 .…... 一 直到 访问 到 最 下 面 一 层 结 点 。 在 同一 层 结 点 中 ， 以 从 左 到 右 的 
顺序 依次 访问 。 我 们 可 以 对 包括 三 又 树 在 内 的 所 有 树 进 行 宽度 优先 裔 
历 。 图 2.5 中 的 三 叉 树 的 宽度 优先 裔 历 的 顺序 是 10、6、14、4、8、12、 
16. 
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左 子 结 点 总 是 小 于 或 等 于 根 结 点 ， 而 右 子 结 点 总 是 大 于 或 等 于 根 结 点 。 
图 2.5 中 的 二 叉 树 束 是 一 柠 二 叉 搜 索 树 。 我 们 可 以 平均 在 O(logn) 的 时 
间 内 根据 数值 在 二 又 搜索 树 中 找到 一 个 结 点 。 二 又 搜 索 树 的 面试 题 有 很 
多 ， 比 如 面试 题 50“ 树 中 两 个 结 点 的 最 低 公 共 祖 先 ” 面试 题 27“ 二 又 搜 
索 树 与 双向 链表 ”。 

二 叉 树 的 另外 两 个 特例 是 堆 和 红 黑 树 。 扒 分 为 最 大 堆 和 最 小 堆 。 在 
最 大 堆 中 根 结 点 的 值 最 大 ， 在 最 小 堆 中 根 结 点 的 值 最 小 。 有 很 多 需要 快 
速 找到 最 大 值 或 者 最 小 值 的 问题 都 可 以 用 堆 来 解决 。 红 黑 树 是 把 树 中 的 
结 点 定义 为 红 、 黑 两 种 颜色 ， 并 通过 规则 确保 从 根 结 点 到 叶 结 点 的 最 长 
路 径 的 长 度 不 超过 最 短路 径 的 两 倍 。 在 C++ 的 STL 中 ，set、mnultiset、 
map、mnultimap 等 数据 结构 都 是 基于 红 黑 树 实 现 的 。 与 堆 和 红 黑 树 相 关 
的 面试 题 ， 请 参考 面试 题 30“ 求 最 小 的 k 个 数字 ”。 


面试 题 6: 重建 二 叉 树 


题目 : 输入 菏 二 又 树 的 前 序 壳 历 和 中 序 明 有 历 的 结果 ， 请 重 
建 出 该 二 义 树 。 假 设 得 入 的 前 序 遇 历 和 中 序 壳 历 的 结案 中 都 不 
含 重 复 的 数字 。 例 如 输入 前 序 过 历 序列 {1,2,4,7,3,5,6,8} 和 中 厅 
名 历 序列 {4,7,2,1,5,3,8,6}， 则 重建 出 图 2.6 所 示 的 二 又 树 并 输出 
它 的 头 结 点 。 二 又 树 结 点 的 定义 如 下 : 
























































struct BinaryTreeNode 


{ 


int m_nValue; 
BinaryTreeNode* m_pLeft; 
BinaryTreeNode* m_pRight; 




















图 2.6 IRH HT AP dd a FP 9 {1,2,4,7,3,5,6,8} 4 H FIN a FP 9) {4,7,2,1,5,3,8,6} He E A) — SC ay 


在 二 又 树 的 前 序 遇 历 序列 中 ， 第 一 个 数字 总 是 树 的 根 结 点 的 值 。 但 
在 中 序 衣 有 历 序 列 中 ， 根 结 点 的 值 在 序列 的 中 间 ， 左 子 树 的 结 点 的 值 位 于 
根 结 点 的 值 的 左边 ， 而 右 子 树 的 结 点 的 值 位 于 根 结 点 的 值 的 右边 。 因 此 
我 们 需要 扫描 中 序 过 历 序列 ， 才 能 找到 根 络 点 的 值 。 

如 图 2.7 所 示 ， 前 序 遍 历 序 列 的 第 一 个 数字 1 束 是 根 结 点 的 值 。 扫 描 
中 序 壳 历 序列 ， 就 能 确定 根 结 点 的 值 的 位 置 。 根 据 中 序 过 有 历 特 点， 在 根 
结 扣 的 值 1 前 面 的 3 个 数字 都 是 左 子 树 结 点 的 值 ， 位 于 1 后 面 的 数字 都 是 
右 子 树 结 点 的 值 。 
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由 于 在 中 序 吉 历 序 列 中 ， 有 3 个 数字 是 左 子 树 结 点 的 值 ， 因 此 左 子 
树 总 共有 3 个 左 子 结 点 。 同 样 ， 在 前 序 遇 历 的 序列 中 ， 根 结 点 后 面 的 3 个 
数字 就 是 3 个 左 子 树 结 点 的 值 ， 再 后 面 的 所 有 数字 都 是 右 子 树 结 扣 的 








值 。 这 样 我 们 束 在 前 序 过 历 和 中 序 过 历 两 个 序列 中 ， 分 别 找 到 了 左右 子 
树 对 应 的 子 序列 。 

既然 我 们 已 经 分 别 找到 了 左 、 右 子 树 的 前 序 过 历 序 列 和 中 序 志 历 序 
列 ， 我 们 可 以 用 同样 的 方法 分 别 去 构建 左右 子 树 。 也 就 是 说 ， 接 下 来 的 
事情 可 以 用 递归 的 方法 去 完成 。 

在 想 清楚 如 何在 前 序 过 历 和 中 序 吉 历 的 序列 中 确定 左 、 右 子 树 的 子 
序列 之 后 ， 我 们 可 以 写 出 如 下 的 递归 代码 : 
BinaryTreeNode* Construct (int* preorder, int* inorder, int length) 


{ 








if (preorder == NULL || inorder == NULL || length <= 0) 
return NULL; 


return ConstructCore(preorder, preorder + length - l, 
inorder, inorder + length - 1); 


} 


BinaryTreeNode* ConstructCore 

( 
int* startPreorder, int* endPreorder, 
int* startInorder, int* endInorder 


// 前 序 遍 历 序列 的 第 一 个 数字 是 根 结 点 的 值 

int rootValue = startPreorder[0]; 
BinaryTreeNode* root = new BinaryTreeNode (); 
root->m_ nValue = rootValue; 

root->m pLeft = root->m_pRight = NULL; 

if (startPreorder == endPreorder) 
{ 

if (startInorder == endInorder 


&& *startPreorder == *startInorder) 
return root; 


throw std: :exception ("Invalid input. 


一 
= 

~~ 

7. 


// 在 中 序 遍 历 中 找到 根 结 点 的 值 

int* rootInorder = startInorder; 

while (rootInorder <= endInorder && *rootInorder != rootValue) 
++ rootiInorder; 


if (rootInorder == endInorder && *rootInorder != rootValue) 
throw std::exception("Invalid input."); 


int leftLength = rootInorder - startInorder; 
int* leftPreorderEnd = startPreorder + leftLength; 
if(leftLength > 0) 
{ 
// 构建 左 子 树 
root->m pLeft = ConstructCore(startPreorder + 1, 
leftPreorderEnd, startInorder, rootInorder = 1); 


} 
if(leftLength < endPreorder - startPreorder) 
{ 


// 构建 右 子 树 
root->m pRight = ConstructCore(leftPreorderEnd + 1, 
endPreorder, rootInorder + 1, endIinorder); 


} 
return root; 


} 

在 函数 ConstructCore 中 ， 我 们 先 根据 前 序 吉 历 序列 的 第 一 个 数字 创 
建 根 结 点 ， 接 下 来 在 中 序 裔 历 序 列 中 找到 根 结 点 的 位 置 ， 这 样 束 能 确定 
左 、 右 子 树 结 点 的 数量 。 在 前 序 壳 历 和 中 序 届 历 的 序列 中 划分 了 左 、 碳 
子 树 结 点 的 值 之 后 ， 我 们 就 可 以 递归 地 调用 函数 ConstructCore， 去 分 别 
构建 它 的 左右 子 树 。 


源 代码 : 
本 题 完 整 的 源 代码 详 见 06_ConstructBinaryTree 项 目 。 
测试 用 例 : 


e 普通 二 又 树 〈 完 全 二 叉 树 ， 不 完全 二 叉 树 ) 。 

e 特殊 二 叉 树 〈 所 有 结 点 都 没有 右 子 结 点 的 二 叉 树 ， 所 有 结 点 都 
没有 左 子 结 点 的 二 叉 树 ， 只 有 一 个 结 点 的 二 叉 树 ) 。 

e 特殊 输入 测试 〈 二 又 树 的 根 结 点 指针 为 NULL、 输 入 的 前 序 通 
历 序列 和 中 序 遍 历 序列 不 匹配 ) 。 




















本 题 考点 : 


o 考 奋 应 聘 者 对 二 又 树 的 前 序 通 历 、 中 序 过 历 的 理解 程度 。 只 有 
对 二 又 树 的 不 同 过 历 算 法 有 了 深刻 的 理解 ， 应 聘 者 才 有 可 能 在 过 历 序列 
中 划分 出 左 、 右 子 树 对 应 的 子 序 列 。 

e 考查 应 聘 者 分 析 复 杂 问 题 的 能 力 。 我 们 把 构建 二 叉 树 的 大 问题 
分 解 成 构建 左 、 右 子 树 的 两 个 小 问题 。 我 们 发 现 小 问题 和 大 问题 在 本 质 
上 是 一 致 的 ， 因 此 可 以 用 递归 的 方式 解决 。 更 多 关于 分 解 复 杂 问 题 的 讨 
论 ， 请 参考 本 书 的 4.4 节 。 


2.3.5” 栈 和 队列 


栈 是 一 个 非常 第 见 的 数据 结构 ， 它 在 计算 机 领域 中 被 广泛 应 用 ， 比 
如 操作 系统 会 给 每 个 线程 创建 一 个 栈 用 来 存储 函数 调用 时 各 个 函数 的 参 
数 、 返 回 地 址 及 临时 变量 等 。 栈 的 特点 十 后 进 先 出 ， 即 最 后 个 压 入 
(push) 栈 的 元 素 会 第 一 个 被 弹出 〈pop) 。 在 面试 题 22“ 栈 的 压 入 、 弹 
出 序列 ?中 ， 我 们 再 详细 分 析 进 栈 和 出 栈 序 列 的 特点 。 

通常 栈 是 一 个 不 考虑 排序 的 数据 结构 ， 我 们 需要 O(n) 时 间 才 能 
找到 栈 中 最 大 或 者 最 小 的 元 系 。 如 果 想 要 在 O〈1) 时 间 内 得 到 栈 的 最 
站 

队列 是 另外 一 种 很 重要 的 数据 结构 。 和 栈 不 同 的 是 ， 队 列 的 特点 是 
先进 移出 ， 即 第 一 个 进入 队列 的 元 系 将 会 第 一 个 出 来 。 在 2.3.4 节 介绍 的 
树 的 宽度 优先 过 历 算法 中 ， 我 们 在 壳 历 茶 一 层 树 的 结 点 时 ， 把 结 点 的 子 
结 点 放 到 一 个 队列 里 ， 以 备 下 一 层 结 点 的 志 历 。 详 细 的 代码 参见 面试 题 
23“ 从 上 到 下 人 所 历 二 叉 树 ”。 

栈 和 队列 虽然 是 特点 针锋相对 的 两 个 数据 结构 ， 但 有 意思 的 是 它们 
却 相 互联 系 。 请 看 面试 题 7“ 用 两 个 栈 实现 队列 *”， 同 时 读者 也 可 以 考虑 
如 何 用 两 个 队列 实现 栈 。 


面试 题 7: 用 两 个 栈 实现 队列 
BA: 用 两 个 栈 实现 一 个 队列 。 队 列 的 声明 如 下 ， 请 实现 


它 的 两 个 函数 appendTail 和 deleteHead， 分 别 完 成 在 队列 尾部 插 
入 结 点 和 在 队列 头 部 删除 结 点 的 功能 。 




















template <typename T> class CQueue 


CQOueue (void); 


~CQueue (void); 


void appendTail (const T& node); 
T deleteHead (); 


stack<T> stackl; 
stack<T> stack2; 
}; 

在 上 述 队 列 的 声明 中 可 以 看 出 ， 一 个 队列 包含 了 两 个 栈 stack1 和 
stack2， 因 此 这 道 题 的 意图 是 要 求 我 们 操作 这 两 个 “先进 后 出 ”的 栈 实 现 
一 个 “先进 先 出 ”的 队列 CQueue。 

我 们 通过 一 个 具体 的 例子 来 分 析 往 该 队列 插入 和 删除 元 素 的 过 程 。 
首先 插入 一 个 元 素 a， 不 妨 先 把 它 插入 到 stack1， 此 时 stack1 中 的 元 素 有 
{a}，stack2 为 空 。 再 压 入 两 个 元 素 b 和 c， 还 是 插入 到 stack1 中 ， 此 时 
stack1 中 的 元 素 有 {a,b,c}， 其 中 c 位 于 栈 顶 ， 而 stack2 仍 然 是 空 的 (如 图 
2.8 (a) 所 示 ) 。 

这 个 时 候 我 们 试 着 从 队列 中 删除 一 个 元 素 。 按 照 队 列 先入 先 出 的 规 
则 ， 由 于 a 比 b、c 先 插入 到 队列 中 ， 最 先 被 删除 的 元 素 应 该 是 a。 元 素 a 
存储 在 stack1 中 ， 但 并 不 在 栈 顶 上 ， 因 此 不 能 直接 进行 删除 。 注 意 到 
stack2 我 们 还 一 直 没 有 使 用 过 ， 现 在 是 让 stack2 发 挥 作用 的 时 候 了 。 如 果 
我 们 把 stack1 中 的 元 素 逐 个 弹出 并 压 入 stack2， 元 素 在 stack2 中 的 顺序 正 
好 和 原来 在 stack1 中 的 顺序 相反 。 因 此 经 过 3 次 弹出 stack1 和 压 入 stack2 操 
作 之 后 ，stack1 为 空 ， 而 stack2 中 的 元 素 是 {c,b,a}， 这 个 时 候 就 可 以 弹出 
stack2 的 栈 顶 a 了。 此 时 的 stack1 为 空 ， 而 stack2 的 元 素 为 {c,b}， 其 中 b 在 
栈 顶 (如 图 2.8 (b) 所 示 ) 。 

如 果 我 们 还 想 继 续 删 除 队 列 的 头 部 应 该 怎么 办 呢 ? 剩 下 的 两 个 元 素 
是 bp 和 c，b 比 c 早 进入 队列 ， 因 此 b 应 该 先 删除 。 而 此 时 b 恰 好 又 在 栈 顶 
上 ， 因 此 直接 弹出 stack2 的 栈 顶 即 可 。 这 次 弹出 操作 之 后 ，stack1 中 仍然 
为 空 ， 而 stack2 为 {c} 〈 如 图 2.8〈c) Aras) 。 

从 上 面 的 分 析 中 我 们 可 以 总 结 出 删除 一 个 元 系 的 步骤 : 当 stack2 中 
不 为 空 时 ， 在 stack2 中 的 栈 顶 元 素 是 最 先进 入 队列 的 元 素 ， 可 以 弹出 。 
如 果 stack2 为 空 时 ， 我 们 把 stack1 中 的 元 素 逐 个 弹出 并 压 入 stack2。 由 于 
先进 入 队列 的 元 素 被 压 到 stack1 的 底 端 ， 经 过 弹出 和 压 入 之 后 就 处 于 
stack2 的 顶端 了 ， 又 可 以 直接 弹出 。 











接 下 来 再 插入 一 个 元 素 d。 我 们 还 是 把 它 压 入 stackl1 (如 图 2.8 (d) 
所 示 ) ， 这 样 会 不 会 有 问题 呢 ? 我 们 考虑 下 一 次 删除 队列 的 头 部 stack2 
不 为 空 ， 直 接 弹出 它 的 栈 顶 元 素 c〈 如 图 2.8 Ce) 所 示 ) 。 而 c 的 确 是 比 d 
先进 入 队列 ， 应 该 在 d 之 前 从 队列 中 删除 ， 因 此 不 会 出 现任 何 了 矛盾。 





stack1 stack2 | stackl stack2 | stackl stack2 


Cc) 删除 队 | (gd) 插入 d | Ce) 删除 队 
列 头 列 头 





图 2.8 用 两 个 栈 模拟 一 个 队列 的 操作 


忆 结 完 每 一 次 在 队列 中 插入 和 删除 操作 的 过 程 之 后 ， 我 们 就 可 以 开 
台 动 手写 代码 了 。 参 考 代码 如 下 : 





template<typename T> void CQueue<T>::appendTail(const T& element) 
{ 
stackl.push (element) ; 


} 


template<typename T> T CQueue<T>: :deleteHead () 


while(stack1.size()>0) 


T& data = stackl.top(); 
stackl.pop(); 
stack2.push (data); 


if(stack2.size() == 0) 


hrow new exception ("queue is empty"); 


源 代码 : 
本 题 完整 的 源 代码 详 见 07_QueueWithTwoStacks 项 目 。 
测试 用 例 : 


o 往 空 的 队列 里 添 加 、 删 除 元 又。 
e 往 非 空 的 队列 里 添加 、 删 除 元 系 。 
e 连续 删除 元 素 直 至 队列 为 空 。 


本 题 考点 : 


© 考 奋 对 栈 和 队列 的 理解 。 

。 考查 写 与 模板 相关 的 代码 的 能 力 。 

© 考查 分 析 复 杂 问 题 的 能 力 。 本 题解 法 的 代码 虽然 只 有 只 有 20 几 
行 代码 ， 但 形成 正确 的 思路 却 不 容易 。 应 聘 者 能 人 否 通过 具体 的 例子 分 析 
问题 ， 通 过 画图 的 手段 把 抽象 的 问题 形象 化 ， 从 而 解决 这 个 相对 比较 复 
杂 的 问题 ， 是 能 耕 顺 利通 过 面试 的 关键 。 








相关 题目 : 


用 两 个 队列 实现 一 个 栈 。 

我 们 通过 一 系列 栈 的 压 入 和 弹出 操作 来 分 析 用 两 个 队列 模拟 一 个 栈 
的 过 程 。 如 图 2.9 (a) 所 示 ， 我 们 先 往 栈 内 压 入 一 个 元 素 a。 由 于 两 个 队 
列 现 在 都 是 空 的 ， 我 们 可 以 选择 把 a 插入 两 个 队列 的 任意 一 个 。 我 们 不 
妨 把 a 插入 queuel。 接 下 来 继续 往 栈 内 压 入 b、c 两 个 元 素 ， 我 们 把 它们 
都 插入 queue1。 这 个 时 候 queuel 包 含 3 个 元 素 a、b 和 c， 其 中 a 位 于 队列 的 
头 部 ，c 位 于 队列 的 尾部 。 

现在 我 们 考虑 从 栈 内 弹出 一 个 元 素 。 根 据 栈 的 后 入 先 出 原则 ， 最 后 
被 压 入 栈 的 c 应 该 最 先 被 弹出 。 由 于 c 位 于 queuel 的 尾部 ， 而 我 们 每 次 只 
能 从 队列 的 头 部 删除 元 素 ， 因 此 我 们 可 以 先 从 queuel 中 依次 删除 元 素 
a、b 并 插入 到 queue2 中 ， 再 从 queue1l 中 删除 元 素 c。 这 融 相 当 于 从 栈 中 弹 
出 元 素 c 了 《如 图 2.9 b) 所 示 ) 。 我 们 可 以 用 同样 的 方法 从 栈 内 弹出 元 
Zeb 〈 如 图 2.9 Cc) 所 示 ) 

接 下 来 我 们 考虑 往 栈 内 压 入 一 个 元 素 d。 此 时 queuel 已 经 有 一 个 元 
素 ， 我 们 就 把 d 插 入 到 queuel 的 尾部 (如 图 2.9(d)〉 Aras) 。 如 果 我 们 再 
从 栈 内 弹出 一 个 元 素 ， 此 时 被 弹出 的 应 该 是 最 后 被 压 入 的 d。 由 于 d 位 于 
queuel 的 尾部 ， 我 们 只 能 先 从 头 删除 queuel 的 元 素 并 插入 到 queue2， 直 
到 在 queue1 中 遇 到 d 再 直接 把 它 删 除 〈 如 图 2.9 Ce) 所 示 ) o 


queuel queue2| queuel queue2| queuel queue2| queuel queue2 | queuel queue2 


(a) RREA | (b) 弹出 c (c) E b (d) Ad 


a. b,c 





图 2.9 用 两 个 队列 模拟 一 个 栈 的 操作 


2.4 算法 和 数据 操作 


和 数据 结构 一 样 ， 考 碍 算法 的 面试 题 也 备 受 面试 官 的 青睐 ， 其 中 排 
序 和 查找 是 面试 时 考查 算法 的 重点 。 在 准备 面试 的 时 候 ， 我 们 应 该 重点 
掌握 二 分 查找 、 归 并 排序 和 快速 排序 ， 做 到 能 随时 正确 、 完 整地 写 出 它 





们 的 代码 。 

有 很 多 算法 都 可 以 用 递归 和 循环 两 种 不 同 的 方式 实现 。 通 利 基 于 递 
归 的 实现 方法 代码 会 比较 简洁 ， 但 性 能 不 如 基于 循环 的 实现 方法 。 在 面 
试 的 时 候 ， 我 们 可 以 根据 题目 的 特点 ， 其 至 可 以 和 面试 官 讨论 选择 合适 
的 方法 编程 。 

位 运算 可 以 看 成 是 一 类 特殊 的 算法 ， 它 是 把 数字 表示 成 二 进 制 之 后 
对 0 和 1 的 操作 。 由 于 位 运算 的 对 象 为 二 进 制 数字 ， 所 以 不 是 很 直观 ， 但 
掌握 它 也 不 难 ， 因 为 总 共 只 有 与 、 或 、 异 或 、 左 移 和 右 移 5 种 位 运算 。 


2.4.1 ”查找 和 排序 


查找 和 排序 都 是 在 程序 设计 中 经 常用 到 的 算法 。 查 找 相对 而 言 较为 
简单 ， 不 外 平顺 序 查 找 、 二 分 查找 、 哈 希 表 查找 和 二 又 排 序 树 查 找 。 在 
面试 的 时 候 ， 不 管 是 用 循环 还 是 用 递归 ， 面 试 官 都 期 待 应 聘 者 能 够 信 手 
持 来 写 出 完整 正确 的 二 分 查找 代码 ， 否 则 可 能 连 继续 面试 的 兴趣 都 没 
有 。 面 试题 8 旋转 数组 的 最 小 数字 ”和 面试 题 38“ 数 字 在 排序 数组 中 出 现 
的 次 数 ” 都 可 以 用 二 分 查找 算法 解决 。 

面试 小 提示 : 

如 果 面 试题 是 要 求 在 排序 的 数组 〈 或 者 部 分 排序 的 数组 ) 中 查找 一 个 数字 或 者 统计 某 个 
数字 出 现 的 次 数 ， 我 们 都 可 以 尝试 用 二 分 查找 算法 。 

哈 希 表 和 二 又 排序 树 查找 的 重点 在 于 考查 对 应 的 数据 结构 而 不 是 算 
法 。 哈 希 表 最 主要 的 优点 是 我 们 利用 它 能 够 在 O (1) 时 间 查 找 某 一 元 
素 ， 是 效率 最 高 的 查找 方式 。 但 其 缺点 是 需要 额外 的 空间 来 实现 哈 希 
表 。 面 试题 35“ 第 一 个 只 出 现 一 次 的 字符 ?就 是 用 哈 希 表 的 特性 来 高 效 查 
找 。 与 二 又 排序 树 查 找 算法 对 应 的 数据 结构 是 二 又 搜索 树 ， 我 们 将 在 面 
试题 24“ 二 又 搜 索 树 的 后 序 遇 历 序列 ”和 面试 题 27“ 二 又 搜索 树 与 双 回 链 
表 ” 中 详细 介绍 二 又 搜 索 树 的 特点 。 

排序 比 查 找 要 复杂 一 些 。 面 试 官 会 经 常 要 求 应 聘 者 比较 插入 排序 、 
冒 泡 排序 、 归 并 排序 、 快 速 排序 等 不 同 算法 的 优 劣 。 强 烈 建议 应 聘 者 在 
准备 面试 的 时 候 ， 一 定 要 对 各 种 排序 算法 的 特点 烂熟 于 胸 ， 能 够 从 额外 
空间 消耗 、 平 均 时 间 复 杂 度 和 最 差 时 间 复 杂 度 等 方面 去 比较 它们 的 优 缺 
点 。 需 要 特别 强调 的 是 ， 很 多 公司 的 面试 官 喜欢 在 面试 环节 中 要 求 应 聘 
者 写 出 快速 排序 的 代码 。 应 聘 者 不 妨 自己 写 一 个 快速 排序 的 函数 并 用 各 
ee ee etapa 
么 区 别 。 

实现 快速 排序 算法 的 关键 在 于 先 在 数组 中 选择 一 个 数字 ， 接 下 来 把 














































































































数组 中 的 数字 分 为 两 部 分 ， 比 选择 的 数字 小 的 数字 移 到 数组 的 左边 ， 比 
选择 的 数字 大 的 数字 移 到 数组 的 右边 。 这 个 函数 可 以 如 下 实现 ; 
int Partition(int data[], int length, int start, int end) 


{ 
if(data == NULL || length <= 0 || start < 0 || end >= length) 
throw new std::exception("Invalid Parameters"); 


int index = RandomInRange(start, end); 
Swap (&data[index], &data[end]); 


int small = start - 1; 
for(index = start; index < end; ++ index) 


{ 
if (data[index] < data[end] ) 


{ 


++ small; 
if (small != index) 
Swap (&data [index], &data[small])j; 


} 


++ small; 
Swap (&data [small], &data[end]); 


return small; 


pki #0 RandomInRangeH KÆ ~ — 7 fEstart#llendZ IAI AY BE ALAL, Pe 
Swap 的 作用 是 用 来 交换 两 个 数字 。 接 下 来 我 们 可 以 用 递归 的 思路 分 别 
0 的 数字 的 左右 两 边 排序 。 下 面 就 是 递归 实现 快速 排序 的 参考 
RIH: 
void QuickSort (int data[], int length, int start, int end) 
{ 


} 





if(start == end) 
return; 


int index = Partition(data, length, start, end); 
if (index > start) 

QuickSort (data, length, start, index - 1); 
if (index < end) 

QuickSort (data, length, index + 1, end); 


对 一 个 长 度 为 n 的 数组 排序 ， 只 需 把 start 设 为 0、 把 end 设 为 n 一 1， 调 
用 函数 QuickSort 即 可 。 

在 前 面 的 代码 中 ， 函 数 Partition 除 了 可 以 用 在 快速 排序 算法 中 ， 还 
可 以 用 来 实现 在 长 度 为 n 的 数组 中 得 找 第 k 大 的 数字 。 面 试题 29“ 数 组 中 
的 数字 ”和 面试 题 30“ 最 小 的 k 个 数 ” 都 可 以 用 这 个 函数 
KHER o 

不 同 的 排序 算法 适用 的 场合 也 不 尽 相 同 。 人 快速 排序 虽然 总 体 的 平均 
效率 是 最 好 的 ， 但 也 不 是 任何 时 候 都 是 最 优 的 算法 。 比 如 数组 本 里 已 经 
排 好 序 了 ， 而 每 一 轮 排序 的 时 候 都 是 以 最 后 一 个 数字 作为 比较 的 标准 ， 
此 时 快速 排序 的 效率 只 有 O m) 。 因 此 在 这 种 场合 快速 排序 就 不 是 最 
优 的 算法 。 在 面试 的 时 候 ， 如 果 面 试 官 要 求实 现 一 个 排序 算法 ， 那 么 应 
聘 者 一 定 要 问 清楚 这 个 排序 应 用 的 环境 是 什么 、 有 哪些 约束 条 件 ， 在 得 
3 00 T EA 
X: 




















面试 官 ， 请 实现 一 个 排序 算法 ， 要 求 时 间 效率 O Cn) 。 

应 聘 者 ， 对 什么 数字 进行 排序 ， 有 多 少 个 数字 ? 

面试 官 :我 们 想 对 公司 所 有 员工 的 年 龄 排序 。 我 们 公司 总 共有 几 万 名 员工 。 

应 聘 者 :也 就 是 说 数字 的 大 小 是 在 一 个 较 小 的 范围 之 内 的 ， 对 吧 ? 

RR: E, EAM. 

应 聘 者 :可 以 使 用 辅助 空间 吗 ? 

面试 官 ， 看 你 用 多 少 辅助 内 存 。 只 允许 使 用 常量 大 小 辅助 空间 ， 不 得 超过 O(n)，。 
在 面试 的 时 候 应 聘 者 不 要 怕 问 面试 官 问题 ， 只 有 多 提问 ， 应 聘 者 才 

有 可 能 明了 面试 官 的 意图 。 在 上 面 的 例子 中 ， 该 应 聘 者 通过 儿 个 问题 就 

卉 清楚 了 需 排 序 的 数字 在 一 个 较 小 的 范围 内 ， 并 且 还 可 以 用 辅助 内 存 。 

知道 了 这 些 限制 条 件 ， 束 不 难 写 出 如 下 的 代码 了 : 




















































































































void SortAges(int ages[], int length) 


{ 
if (ages == NULL || length <= 0) 
return; 


const int oldestAge = 99; 
int timesOfAge[oldestAge + 1]; 


for(int i= 0; i <= oldestAge; ++ i) 
timesOfAge[i] = 0; 

for(int. 1 = OF; © < length; ++. 1) 

{ 


throw new std::exception("age out of range."); 


++ timesOfAge [age]; 


int index = 0; 
for{tint: i = 0 


{ 
for(int j = j < timesOfAge [i] 
{ 
ages [index] = i; 
++ index; 
} 
} 


} 

公司 员工 的 年 龄 有 一 个 范围 。 在 上 面 的 代码 中 ， 人 允许 的 范围 是 0 一 
99 岁 。 数 组 timesOfAge 用 来 统计 每 个 年 龄 出 现 的 次 数 。 某 个 年 龄 出 现 了 
多 少 次 ， 就 在 数组 ages 里 设置 几 次 该 年 龄 ， 这 样 就 相当 于 给 数组 ages 排 
nen 该 方法 用 长 度 100 的 整数 数组 作为 辅助 空间 换 来 了 O Cn) 的 时 间 
MR 


面试 题 8; Te Fe BLA BY te“ Be 
题目 :把 一 个 数组 最 开始 的 若干 个 元 素 搬 到 数组 的 末尾 ， 


我 们 称 之 为 数组 的 旋转 。 输 入 一 个 递增 排序 的 数组 的 一 个 旋 
转 ， 输 出 旋转 数组 的 最 小 元 素 。 例 如 数组 {3,4,5,1,2} 为 





{1,2,3,4,5} 的 一 个 旋转 ， 该 数组 的 最 小 值 为 1。 

这 着 题 最 直观 的 解法 并 不 难 ， 从 头 到 尾 过 历数 组 一 次 ， 我 们 就 能 找 
出 最 小 的 元 了 淼 。 这 种 思路 的 时 间 复 杂 度 显然 是 O Cn) 。 但 是 这 个 思路 
没有 利用 输入 的 旋转 数组 的 特性 ， 肯 定 达 不 到 面试 官 的 要 求 。 

我 们 注意 到 旋转 之 后 的 数组 实际 上 可 以 划分 为 两 个 排序 的 子 数 组 ， 
而 且 前 面 的 子 数组 的 元 素 都 大 于 或 者 等 于 后 面子 数组 的 元 系 。 我 们 还 注 
意 到 最 小 的 元 素 刚 好 是 这 两 个 子 数组 的 分 界线 。 在 排序 的 数组 中 我 们 可 
以 用 二 分 查找 法 实现 O(logn〉 的 奋 找 。 本 题 给 出 的 数组 在 一 定 程 度 上 
征 排序 的 ， 因 此 我 们 可 以 试 着 用 二 分 碍 找 法 的 思路 来 寻找 这 个 最 小 的 元 


和 二 分 查找 法 一 样 ， 我 们 用 两 个 指针 分 别 指 问 数 组 的 第 一 个 元 素 和 
最 后 一 个 元 素 。 按 照 题 日 中 旋转 的 规则 ， 第 一 个 元 素 应 该 是 大 于 或 者 等 
于 最 后 一 个 元 素 的 〈 这 其 实 不 完全 对 ， 还 有 特例 ， 后 面 再 加 以 讨论 ) 。 

接着 我 们 可 以 找到 数组 中 间 的 元 素 。 如 果 该 中 间 元 素 位 于 前 面 的 递 
增 子 数组 ， 那 么 它 应 该 大 于 或 者 等 于 第 一 个 指针 指 问 的 元 素 。 此 时 数组 
中 最 小 的 元 素 应 该 位 于 该 中 间 元 素 的 后 面 。 我 们 可 以 把 第 一 个 指针 指 问 
该 中 间 元 系 ， 这 样 可 以 缩小 寻找 的 范围 。 移 动 之 后 的 第 一 个 指针 仍然 位 
于 前 面 的 递增 子 数 组 之 中 。 

同样 ， 如 果 中 间 元 素 位 于 后 面 的 递增 子 数组 ， 那 么 它 应 该 小 于 或 者 
等 于 第 二 个 指针 指向 的 元 素 。 此 时 该 数组 中 最 小 的 元 素 应 该 位 于 该 中 间 
元 系 的 前 面 。 我 们 可 以 把 第 二 个 指针 指 同 该 中 间 元 素 ， 这 样 也 可 以 缩小 
寻找 的 范围 。 移 动 之 后 的 第 二 个 指针 仍然 位 于 后 面 的 递增 子 数组 之 中 。 

不 管 是 移动 第 一 个 指针 还 是 第 二 个 指针 ， 碍 找 苑 围 都 会 缩小 到 原来 
的 一 半 。 接 下 来 我 们 再 用 更 新 之 后 的 两 个 指针 ， 重 复 做 新 一 轮 的 查找 。 

按照 上 述 的 思路 ， 第 一 个 指针 总 是 指 癌 前 面 递增 数组 的 元 素 ， 而 第 
二 个 指针 总 是 指 问 后 面 北 增 数组 的 元 素 。 最 终 第 一 个 指针 将 指 同 前 面子 
数组 的 最 后 一 个 元 素 ， 而 第 二 个 指针 会 指 同 后 面子 数组 的 第 一 个 元 素 。 
也 束 是 它们 最 终 会 指 问 两 个 相 邻 的 元 素 ， 而 第 二 个 指针 指 同 的 刚好 是 最 
小 的 元 素 。 这 就 是 循环 结束 的 条 件 。 

以 前 面 的 数组 {3,4,5,1,2} 为 例 ， 我 们 先 把 第 一 个 指针 指 同 第 0 个 元 
素 ， 把 第 二 个 指针 指向 第 4 个 元 素 〈 如 图 2.10 (a) 所 示 ) 。 位 于 两 个 指 
针 中 间 《〈 在 数组 中 的 下 标 是 2) 的 数字 是 5， 它 大 于 第 一 个 指针 指 回 的 数 
字 。 因 此 中 间 数 字 5 一 定位 于 第 一 个 递增 子 数组 中 ， 并 且 最 小 的 数字 一 
定位 于 它 的 后 面 。 因 此 我 们 可 以 移动 第 一 个 指针 让 它 指向 数组 的 中 间 
(图 2.10 (b) 所 示 ) 。 

此 时 位 于 这 两 个 指针 中 间 (在 数组 中 的 下 标 是 3) 的 数字 是 1， 它 小 
于 第 二 个 指针 指 同 的 数字 。 因 此 这 个 中 间 数 字 1 一 定位 于 第 二 个 递增 字 




































































数组 中 ， 并 且 最 小 的 数字 一 定位 于 它 的 前 面 或 者 它 自己 就 是 最 小 的 数 
字 。 因 此 我 们 可 以 移动 第 二 个 指针 指 癌 两 个 指针 中 间 的 元 素 即 下 标 为 3 
的 元 素 〈 如 图 2.10 (c) 所 示 ) 。 


P1 P2 











(b) 


Ti ko 
图 2.10 ”在 数组 {3,4,5,1,2} 中 查找 最 小 值 的 过 程 


TE: 旋转 数组 中 包含 两 个 递增 排序 的 子 数组 ， 有 阴影 背景 的 是 第 二 个 子 数 组 。 (a) 把 P1 
指向 数组 的 第 一 个 数字 ，P2 指 向 数组 的 最 后 一 个 数字 。 由 于 P1 和 P2 中 间 的 数字 5 大 于 P1 指 向 的 
数字 ， 中 间 的 数字 在 第 一 个 子 数组 中 。 一 步 把 P1 指 向 中 间 的 数字 。 (b) P1 和 P2 中 间 的 数字 1 
小 于 P2 指 向 的 数字 ， 中 间 的 数字 在 第 二 个 子 数 组 中 。 下 一 步 把 P2 指 向 中 间 的 数字 。 Cc) PA 
P2 指 向 两 个 相 邻 的 数字 ， 则 了 2 指向 的 是 数组 中 的 最 小 数字 。 


此 时 两 个 指针 的 距离 是 1， 表 明 第 一 个 指针 已 经 指向 了 第 一 个 递增 
子 数组 的 末尾 ， 而 第 二 个 指针 指 问 第 二 个 递增 子 数 组 的 开头 。 第 二 个 子 
数组 的 第 一 个 数字 不是 最 小 的 数字 ， 因 此 第 二 个 指针 指向 的 数字 就 是 我 
MERKER. 

基于 这 个 思路 ， 我 们 可 以 写 出 如 下 代码 : 
























































int Min(int* numbers, int length) 
{ 
if (numbers == NULL || length <= 0) 


throw new std::exception("Invalid parameters") ; 


int indexl = 0; 
int index2 = length - 
int indexMid = index1; 
while (numbers [indexl] >= numbers [index2]) 
{ 

if (index2 - indexl == 1) 


{ 


-r 


D 


indexMid = index2; 


break; 

} 

indexMid = (indexl + index2) 4 

if (numbers [indexMid] >= numbers [index1]) 
indexl = indexMid 


index2 = indexMid; 


} 


return numbers [indexMid]; 


前 面 我 们 提 到 在 旋转 数组 中 ， 由 于 是 把 递增 排序 数组 前 面 的 若干 个 
数字 搬 到 数组 的 后 面 ， 因 此 第 一 个 数字 总 是 大 于 或 者 等 于 最 后 一 个 数 
字 。 但 按照 定义 还 有 一 个 特例 :如 果 把 排序 数组 的 前 面 的 0 个 元 素 搬 到 
最 后 面 ， 即 排序 数组 本 身 ， 这 仍然 是 数组 的 一 个 旋转 ， 我 们 的 代码 需要 
支持 这 种 情况 。 此 时 ， 数 组 中 的 第 一 个 数字 就 是 最 小 的 数字 ， 可 以 直接 
返回 。 这 就 是 在 上 面 的 代码 中 ， 把 indexMid 初 始 化 为 iIndex1 的 原因 。 一 
且 发 现 数组 中 第 一 个 数字 小 于 最 后 一 个 数字 ， 表 明 该 数组 是 排序 的 ， 就 
可 以 直接 返回 第 一 个 数字 了 。 

上 述 代码 是 否 就 完美 了 呢 ? 面试 官 会 告诉 我 们 其 实 不 然 。 他 将 提示 
我 们 再 仔细 分 析 下 标 为 iIndex1 和 index2 (index1 和 index2 分 别 和 图 中 P1 和 
P2 相 对 应 ) 的 两 个 数 相同 的 情况 。 在 前 面 的 代码 中 ， 当 这 两 个 数 相同 ， 
并 且 它 们 中 间 的 数字 〈 即 indexMid 指 向 的 数字 ) 也 相同 时 ， 我 们 把 
indexMid 赋 值 给 了 index1， 也 就 是 认为 此 时 最 小 的 数字 位 于 中 间 数 字 的 
后 面 。 是 不 是 一 定 这 样 ? 

我 们 再 来 看 一 个 例子 。 数 组 {1,0,1,1,1} 和 数组 {1,1,1,0,1} 都 可 以 看 成 
是 递增 排序 数组 {0,1,1,1,1} 的 旋转 ， 图 2.11 分 别 画 出 它们 由 最 小 数字 分 
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隅 开 的 两 个 子 数组 。 
E 
Tp 1 P1 P2 


图 2.11 ”数组 {0,1,1,1,1} 的 两 个 旋转 {1,0,1,1,1} 和 {1,1,1,0,1} 


: 在 这 两 个 数组 中 ， 第 一 个 数字 、 最 后 一 个 数字 和 中 间 数 字 都 是 1， 我 们 无 法 确定 中 间 
KFIR 于 第 一 个 递增 子 数组 还 是 属于 第 二 个 递增 子 数组 。 第 二 个 子 数组 用 灰色 背景 表示 。 


在 这 两 种 情况 中 ， 第 一 个 指针 和 第 二 个 指针 指向 的 数字 都 是 1， 并 
且 两 个 指针 中 间 的 数字 也 是 1， 这 3 个 数字 相同 。 在 第 一 种 情况 中 ， 中 间 
数字 〈 下 标 为 2) 位 于 后 面 的 子 数 组 ， 在 第 二 种 情况 中 ， 中 间 数 字 (下 
标 为 2) 位 于 前 面 的 子 数组 中 。 因 此 ， 当 两 个 指针 指向 的 数学 及 它们 中 
间 的 数字 三 者 相同 的 时 候 ， 我 们 无 法 判断 中 间 的 数字 是 位 于 前 面 的 子 数 
组 中 还 是 后 面 的 子 数 组 中 ， 也 就 无 法 移动 两 个 指针 来 缩小 查找 的 范围 。 
此 时 ， 我 们 不 得 不 采用 顺序 碍 找 的 方法 。 
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int Min(int* numbers, int length) 
{ 
if (numbers == NULL |] length <= 0) 
throw new std::exception("Invalid parameters"); 


int indexl = 0; 
int index2 = length - 1; 
int indexMid = indexl; 
while (numbers[index1] >= numbers [index2]) 
{ 
if(index2 - indexi == 1) 
{ 
indexMid = index2; 
break; 
} 


indexMid = (indexl + index2) / 2; 


// 如 果 下 标 为 indexl. index2 和 indexMid 指向 的 三 个 数字 相等 ， 
// 则 只 能 顺序 查找 
if (numbers [index1] == numbers [index2] 

&& numbers[indexMid] == numbers [index1]) 

return MinInOrder(numbers, indexl, index2); 


if (numbers [indexMid] >= numbers [index1]) 
indexl = indexMid; 

else if (numbers[indexMid] <= numbers[index2]) 
index2 = indexMid; 


} 


return numbers [indexMid]; 


} 


int MinInOrder(int* numbers, int indexl, int index2) 


{ 
int result = numbers [indexl1]; 
for(int i = indexl + 1; i <= index2; ++i) 
{ 
if (result > numbers[i]) 
result = numbers [i]; 


} 


return result; 


源 代码 : 
本 题 完 整 的 源 代码 详 见 08_MinNumberInRotatedArray 项 目 。 
测试 用 例 : 


e 功能 测试 〈 输 入 的 数组 是 升序 排序 数组 的 一 个 旋转 ， 数 组 中 有 
重复 数字 或 者 没有 重复 数字 ) 。 

o 边界 值 测试 〈 输 入 的 数组 是 一 个 升序 排序 的 数组 、 只 包含 一 个 
数字 的 数组 ) 。 

e 特殊 输入 测试 〈 输 入 NULL 指 针 ) 。 


本 题 考点 : 


© 考 奋 对 二 分 碍 找 的 理解 。 本 题 变 换 了 二 分 碍 找 的 条 件 ， 输 入 的 
数组 不 是 排序 的 ， 而 是 排序 数组 的 一 个 旋转 。 这 要 求 我 们 对 二 分 碍 找 的 
过 程 有 深刻 的 理解 。 

。 考查 沟通 学 习 能 力 。 本 题 面试 官 提出 了 一 个 新 的 概念 : 数组 的 
旋转 。 我 们 要 在 很 短 时 间 内 学 习 理解 这 个 新 概念 。 在 面试 过 程 中 如 果 面 
试 官 沁 出 新 的 役 售 ， 我 们 可 以 主动 和 面 读 官 沟通 ， 多 门 儿 个 问题 把 入 仿 
弄 清楚 。 

o 考 奋 思维 的 全 面 性 。 排 序数 组 本 吴 是 数组 旋转 的 一 个 特例 。 忆 
外 ， 我 们 要 考虑 到 数组 中 有 相同 数字 的 特例 。 如 果 不 能 很 好 地 处 理 这 些 
特例 ， 束 很 难 写 出 让 面试 官 满意 的 完美 代码 。 


2.4.2 VARIA 


如 果 我 们 需要 重复 地 多 次 计算 相同 的 问题 ， 通 常 可 以 选择 用 递归 或 
者 循环 两 种 不 同 的 方法 。 递 归 是 在 一 个 函数 的 内 部 调用 这 个 函数 目 身 。 
而 循环 则 是 通过 设置 计算 的 初始 值 及 终止 条 件 ， 在 一 个 范围 内 重复 运 
算 。 比 如 求 1 十 2 十 .…. 十 n， 我 们 可 以 用 递归 或 者 循环 两 种 方式 求 出 结 
果 。 对 应 的 代码 如 下 : 

















int AddFromlToN Recursive (Int n) 


{ 

return: i <= 0 7 0 $ nh + AddFromlToN Recursive (n Sere OY IG 
} 
int AddFrom1ToN Iterative(int n) 
{ 

int result = 

FOr(Int L= 45) i ten EOD 

result += 1; 


通常 递归 的 代码 会 比较 简洁 。 在 上 面 的 例子 里 ， 递 归 的 代码 只 有 一 
个 语句 ， 而 循环 则 需要 4 个 语句 。 在 树 的 前 序 、 中 序 、 后 序 遍 历 算法 的 
代码 中 ， 递 归 的 实现 明显 要 比 循环 简单 得 多 。 在 面试 的 时 候 ， 如 果 面 试 
官 没有 特别 的 要 求 ， 应 聘 者 可 以 尽量 多 采用 递归 。 

面试 小 提示 : 

通常 基于 递归 实现 的 代码 比 基 于 循环 实现 的 代码 要 简洁 很 多 ， 更 加 容易 实现 。 如 果 面 试 
官 没有 特殊 要 求 ， 应 聘 者 可 以 优先 采用 递归 的 方法 编程 。 

递归 虽然 有 简洁 的 优点 ， 但 它 同 时 也 有 显著 的 缺点 。 递 归 由 于 是 函 
数 调用 自身 ， 而 函数 调用 是 有 时 间 和 空间 的 消耗 的 : 每 一 次 函数 调用 ， 
都 需要 在 内 存 栈 中 分 配 空 间 以 保存 参数 、 返 回 地 址 及 临时 变量 ， 而 且 往 
栈 里 压 入 数据 和 弹出 数据 都 需要 时 间 。 这 就 不 难 理解 上 述 的 例子 中 递归 
实现 的 效率 不 如 循环 。 

另外 ， 递 归 中 有 可 能 很 多 计算 都 是 重复 的 ， 从 而 对 性 能 带 来 很 大 的 
负面 影响 。 递 归 的 本 质 是 把 一 个 问题 分 解 成 两 个 或 者 多 个 小 问题 。 如 果 
多 个 小 问题 存在 相互 重 车 的 部 分 ， 那 么 就 存在 重复 的 计算 。 在 面试 题 
9“ 斐 波 那 契 数列 ”及 面试 题 43“n 个 般 子 的 点 数 ” 中 我 们 将 详细 地 分 析 递 归 
和 循环 的 性 能 区 别 。 

除了 效率 之 外 ， 递 归还 有 可 能 引起 更 严重 的 问题 : 调用 栈 溢出 。 前 
面 分 析 中 提 到 需要 为 每 一 次 函数 调用 在 内 存 栈 中 分 配 空间 ， 而 每 个 进程 
的 栈 的 容量 是 有 限 的 。 当 递归 调用 的 层级 太 多 时 ， 就 会 超出 栈 的 容量 ， 
从 而 导致 调用 栈 溢 出 。 在 上 述 例子 中 ， 如 果 输 入 的 参数 比较 小 ， 如 10， 
它们 都 能 返回 结果 55。 但 如 果 输 入 的 参数 很 大 ， 如 5000， 那 么 递归 代码 
在 运行 的 时 候 就 会 出 错 ， 但 运行 循环 的 代码 能 得 到 正确 的 结果 
12502500. 



























































TAO: SEW AL BHI 


题目 一 : 写 一 个 图 数 ， 输 入 n， 求 斐 波 那 契 (Fibonacci) 
数列 的 第 Bn. EVR ABSA Ee CU F: 
A n=Q 
fm) = n=1 


J Zz} n>1 
效率 很 低 的 解法 ， 挑 剔 的 面试 官 不 会 喜欢 
很 多 C 语 言 教 科 书 在 讲述 递归 函数 的 时 候 ， 部 会 用 Fibonacci 作 为 例 


子 ， 因 此 很 多 应 聘 者 对 这 道 题 的 递归 解法 部 很 熟悉。 他 们 看 到 这 道 题 的 
时 候 心 中 会 处 不 住 一 阵 鳃 言 ， 因 为 他 们 能 很 快 写 出 如 下 代码 : 





long long Fibonacci(unsigned int n) 
{ 
if(n <= Q) 
return 0; 
if(n == 1) 
return 1; 
return Fibonacci(n - 1) + Fibonacci(n - 2); 


} 

我 们 的 教科 书 上 反复 用 这 个 问题 来 讲解 递归 函数 ， 并 不 能 说 明 递 归 
的 解法 最 适合 这 道 题目 。 面 试 官 会 提示 我 们 上 述 递归 的 解法 有 很 严重 的 
效率 问题 并 要 求 我 们 分 析 原 因 。 

我 们 以 求解 { (10) 为 例 来 分 析 递 归 的 求解 过 程 。 想 求 得 f〈10) ， 
需要 先 求 得 f (9) Alf (8) 。 同 样 ， 想 求 得 f (9) ， 需 要 先 求 得 f (8) 
METI sa FeAl yay DL ANE 吉 构 来 表示 这 种 依赖 关系 ， 如 图 2.12 所 
示 。 








图 2.12 ”基于 递归 求 斐 波 那 契 数列 的 第 10 项 的 调用 过 程 


我 们 不 难 发 现在 这 棵 树 中 有 很 多 结 点 是 重复 的 ， 而 且 重 复 的 结 点 数 
会 随 着 n 的 增 大 而 急剧 增加 ， 这 意味 计算 量 会 随 着 n 的 增 大 而 急剧 增 大 。 
事实 上 ， 用 递归 方法 计算 的 时 间 复 杂 度 是 以 n 的 指数 的 方式 递增 的 。 读 
者 不 妨 求 Fibonacci 的 第 100 项 试 试 ， 感 受 一 下 这 样 递归 会 慢 到 什么 程 
EE. 


面试 官 期 待 的 实用 解法 


其 实 改进 的 方法 并 不 复 人 本。 上 述 递 归 代 码 之 所 以 慢 是 因为 重复 的 计 
算 太 多 ， 我 们 只 要 想 办 法 避免 重复 计算 融 行 了 。 比 如 我 们 可 以 把 己 经 得 
到 的 数列 中 间 项 保存 起 来 ， 如 宋 下 次 需要 计算 的 时 候 我 们 先 碍 找 一 个， 
如 果 前 面 己 经 计算 过 就 不 用 再 重复 计算 了 。 

更 简单 的 办 法 是 从 下 往 上 计算 ， 首 先 根据 f (0〉 和 f C1) 算出 
f (2〉 ， 再 根据 f (1) Mf (2) 算出 f (3) ...... 依 此 类 推 就 可 以 算出 第 n 
项 了 。 很 容易 理解 ， 这 种 思路 的 时 间 复 杂 度 是 DO Cn) 。 实 现代 码 如 
WP: 














long long Fibonacci (unsigned n) 
{ 
int result[2] = {0, 1}; 
if(n < 2) 
return result([n]; 


long long fibNMinusOne = 1; 


long long fibNMinusTwo Gs 
long long fibN = 0; 
for(unsigned int i = 2; i <= n; ++ i) 


{ 
fibN = fibNMinusOne + fibNMinusTwo; 


fibNMinusTwo = fibNMinusOne; 
fibNMinusOne = fibN; 
} 


return fibN; 


} 
时 间 复 杂 度 O dogn) 但 不 够 实用 的 解法 


通常 面试 到 这 里 也 就 差不多 了 ， 尽 管 我 们 还 有 比 这 更 快 的 
O Cogn) 算法 。 由 于 这 种 算法 需要 用 到 一 个 很 生父 的 数学 公式 ， 因 此 
很 少 有 面试 官 会 要 求 我 们 掌握 。 不 过 以 防 不 时 之 需 ， 我 们 还 是 简要 介绍 
一 下 这 种 算法 。 

在 介绍 这 种 方法 之 前 ， 我 们 先 介绍 一 个 数学 公式 : 


fin) f(n-l) 1 | n-l 
aeara] s | 
这 个 公式 用 数学 归纳 法 不 难 证 明 ， 感 兴趣 的 读者 不 妨 自 己 证 明 一 
1 1" 
Pe ATATA 我 们 只 需要 求 得 矩阵 [1 al 即 可 得 到 f Cn) 。 现 
: ， 
eft aan Arta sea ela 巴 的 乘 方 。 如 果 只 是 简单 地 从 0 开始 循环 ， 
n 次 方 需 要 n 次 运算 ， 那 其 时 间 复 杂 度 仍然 是 DO Cn) ， 并 不 比 前 面 的 方 
法 快 。 但 我 们 可 以 考虑 乘 方 的 如 下 性 质 : 
a" = fea n 为 个 数 


> Me AS ee 
1" 1/2 in 2 nh D 








从 上 面 的 公式 我 们 可 以 看 出 ， 我 们 想 求 得 n 次 方 ， 就 要 先 求 得 n/2 次 
方 ， 再 把 nD/2 次 方 的 结果 平方 一 下 即 可 。 这 可 以 用 递归 的 思路 实现 。 

由 于 很 少 有 面试 官 要 求 编程 实现 这 种 思路 ， 本 书 中 不 再 列 出 完整 的 
代码 ， 感 兴趣 的 读者 请 参考 附 市 的 源 代码 。 不 过 这 种 基于 加 归 用 
O dogn) 的 时 间 求 得 n 次 方 的 算法 却 值得 我 们 重视 。 我 们 在 面试 题 
11“ 数 值 的 整数 次 方 ” 中 再 详细 讨论 这 种 算法 。 


解法 比较 


用 不 同 的 方法 求解 裴 波 那 契 数 列 的 时 间 效 率 大 不 相同 。 第 一 种 基于 
递归 的 解法 虽然 直观 但 时 间 效 率 很 低 ， 在 实际 软件 开发 中 不 会 用 这 种 方 
法 ， 也 不 可 能 得 到 面试 官 的 青睐 。 第 二 种 方法 把 递归 的 算法 用 循环 实 
现 ， 极 大 地 提高 了 时 间 效 率 。 第 三 种 方法 把 求 辈 波 那 契 数列 转换 成 求 矩 
阵 的 乘 方 ， 是 一 种 很 有 创意 的 算法 。 虽 然 我 们 可 以 用 O〈logn) REE 
阵 的 n 次 方 ， 但 由 于 隐 含 的 时 间 常 数 较 大 ， 很 少 会 有 软件 及 用 这 种 算 
法 。 力 外 ， 实 现 这 种 解法 的 代码 也 很 复杂 ， 不 太 适 用 面试 。 因 此 第 三 种 
方法 不 是 一 种 实用 的 算法 ， 不 过 应 聘 者 可 以 用 它 来 展示 目 己 的 知识 面 。 

除了 面试 官 直 接 要 求 编程 实现 裴 波 那 韶 数列 之 外 ， 还 有 不 少 面试 题 
BY Bre SEAS BUA IMA, EGU: 

题目 二 : 一 只 青蛙 一 次 可 以 跳 上 1 级 台阶 ， 也 可 以 跳 上 2 
级 。 求 该 青蛙 跳 上 一 个 n 级 的 台阶 总 共有 多 少 种 跳 法 。 

首先 我 们 考虑 最 简单 的 情况 。 如 果 只 有 1 级 人 台阶 ， 那 显然 只 有 一 种 
跳 法 。 如 果 有 2 级 台阶 ， 那 就 有 两 种 跳 的 方法 了 : 一 种 是 分 两 次 跳 ， 
次 跳 1 级 ; 为 外 一 种 就 是 一 次 跳 2 级 。 

接 独 我 们 再 来 讨论 一 般 情 况 。 我 们 把 级 合 阶 时 的 跳 法 看 成 是 n 的 函 
数 ， 记 为 f Cn) 。 当 n>2 时 ， 第 一 次 跳 的 时 候 束 有 两 种 不 同 的 选择 : 一 
古 第 一 次 只 跳 1 级 ， 此 时 跳 法 数目 等 于 后 面 剩 下 的 n 一 1 级 合 阶 的 跳 法 数 
目 ， 即 为 f Cn 一 1) ; 为 外 一 种 选择 是 第 一 次 跳 2 级 ， 此 时 跳 法 数目 等 于 
后 面 剩 下 的 n 一 2 级 人 台阶 的 跳 法 数目 ， 即 为 f{ Cn 一 2) 。 因 此 n 级 台阶 的 不 
同 跳 法 的 总 数 f (a) =f Cn 一 1) +f Cn 一 2) 。 分 析 到 这 里 ， 我 们 不 难看 
出 这 实际 上 束 是 辈 波 那 契 数列 了 。 


源 代码 : 
本 题 完 整 的 源 代 码 详 见 09_Fibonacci 项 目 。 
测试 用 例 : 









































e 功能 测试 〈 如 输入 3、5、10 等 ) 。 
e 边界 值 测试 (如 输入 0、1、2) 。 
e 性 能 测试 (输入 较 大 的 数字 ， 如 40、50、100 等 ) 。 


本 题 考点 : 


e 考 香 对 递归 、 循 环 的 理解 及 编码 能 

o 考 奋 对 时 间 复 杂 上 度 的 分 析 能 力 。 

e 如 宋 面 试 官 采 用 的 是 青蛙 跳台 阶 的 问题 ， 那 同时 还 在 考查 应 聘 
者 的 数学 建 模 能 


本 题 扩展 : 


在 青蛙 跳台 阶 的 问题 中 ， 如 果 把 条 件 改 成 : 一 只 青蛙 一 次 可 以 跳 上 
1 级 人 台阶， 也 可 以 跳 上 2 级 ..……… 它 也 可 以 跳 上 n 级 ， 此 时 该 青蛙 跳 上 一 个 n 
级 的 台阶 总 共有 多 少 种 跳 法 ? 我 们 用 数学 归纳 法 可 以 证 明 f m) =2"". 


相关 题目 : 


我 们 可 以 用 2x1 (图 2.13 的 左边 ) 的 小 矩形 横着 或 者 竖 着 去 覆盖 更 
大 的 矩形 。 请 问 用 8 个 2x1 的 小 矩形 无 重 早 地 和 窗 亲 一 个 2x8 的 大 算 形 (图 
2.13 的 右边 ) ， 总 共有 多 少 种 方法 ? 


d EHHH 


图 2.13 ”一 个 2x1 的 矩形 和 2x8 的 矩形 


我 们 先 把 2x8 的 履 盖 方法 记 为 f{ (8) 。 用 第 一 个 1x2 小 矩形 去 覆盖 大 
窍 形 的 最 左边 时 有 两 个 选择 ， 竖 着 放 或 者 横着 放 。 当 竖 着 放 的 时 候 ， 碳 
边 还 剩 下 2x7 的 区 域 ， 这 种 情形 下 的 履 盖 方法 记 为 f{ (7) 。 接 下 来 考虑 
横着 放 的 情况 。 当 1x2 的 小 矩形 横着 放 在 左上 角 的 时 候 ， 左 下 角 必 须 和 
横着 放 一 个 1x2 的 小 和 矩形， 而 在 右边 还 还 剩 下 2x6 的 区 域 ， 这 种 情形 下 的 
履 盖 方法 记 为 f (6) ， 因 此 f (8) =f (7) +f (6) 。 此 时 我 们 可 以 看 
tH, 这 仍然 是 斐 波 那 契 RBI o 


2.4.3 ”位 运算 


位 运算 是 把 数字 用 二 进 制 表示 之 后 ， 对 每 一 位 上 0 或 者 1 的 运算 。 二 
进 制 及 其 位 运算 是 现代 计算 机 学 科 的 基石 ， 很 多 底层 的 扩 术 都 离 不 开 位 



































运算 ， 因 此 位 运算 相关 的 题目 也 经 常 出 现在 面试 中 。 由 于 我 们 在 日 常生 
活 中 习惯 了 十 进 制 ， 很 多 人 看 到 二 进 制 及 位 运算 都 觉得 很 难 适应 。 

理解 位 运算 的 第 一 步 是 理解 二 进 制 。 二 进 制 是 指数 字 的 每 一 位 都 是 
0 或 者 1。 比 如 十 进 制 的 2 转化 为 二 进 制 之 后 是 10， 而 十 进 制 的 10 转 换 成 
二 进 制 之 后 是 1010。 在 程序 员 圈 子 里 有 一 个 流传 了 很 久 的 笑话 ， 说 世界 
上 有 10 种 人 ， 一 种 人 知道 二 进 制 ， 而 另 一 种 人 不 知道 二 进 制 ..………. 

除了 二 进 制 ， 我 们 还 可 以 把 数字 表示 成 其 他 进 制 ， 比 如 表示 时 间 分 
秒 的 六 十 进 制 等 。 针 对 不 太 熟 悉 的 进 制 ， 已 经 出 现 了 不 少 很 有 意思 的 面 
试题 。 比 如 : 

在 Excel 2003 中 ， 用 A 表示 第 1 列 ，B 表 示 第 2 列 ...... Z 表 示 第 26 列 ， 
AA 表 示 第 27 列 ，AB 表 示 第 28 列 ...... 以 此 类 推 。 请 写 出 一 个 函数 ， 输 入 
用 字母 表示 的 列 号 编码 ， 输 出 它 是 第 几 列 。 

这 是 一 道 很 新 颖 的 关于 进 制 的 题目 ， 其 本 质 是 把 十 进 制 数 字 用 A 一 
Z 表 示 成 二 十 六 进 制 。 如 果 想 到 这 一 点 ， 解 决 这 个 问题 就 不 难 了 。 

其 实 二 进 制 的 位 运算 并 不 是 很 难 掌握 ， 因 为 位 运算 总 共 只 有 五 种 运 
算 : 与 、 或 、 异 或 、 左 移 和 右 移 。 与 、 或 和 异 或 运算 的 规律 我 们 可 以 用 
表 2.1 总 结 如 下 : 


























表 2.1 与 、 或 、 异 或 的 运算 规律 


5(&) 0&0=0 1&0=0 0&1=0 


或 (1) 010=0 110=] 011=] 





异 或 (人 ^) 0^0=0 1A0=1 0^1=1 


左 移 运 算 符 m<<n 表 示 把 m 左 移 n 位 。 左 移 n 位 的 时 候 ， 最 左边 的 n 位 

















将 被 丢弃 ， 同 时 在 最 右边 补 上 n 个 0。 比 如 : 

00001010<<2=00101000 

10001010<<3=01010000 

右 移 运算 符 m>>n 表 示 把 囊 右 移 n 位 。 右 移 n 位 的 时 候 ， 最 右边 的 n 位 
将 被 丢弃 。 但 右 移 时 处 理 最 左边 位 的 情形 要 稍微 复杂 一 点 。 如 果 数 字 是 
个 无 符号 数值 ， 则 用 0 填补 最 左边 的 n 位 。 如 果 数 字 是 一 个 有 符号 数 
值 ， 则 用 数字 的 符号 位 填补 最 左边 的 n 位 。 也 就 是 说 如 果 数 字 原 先是 一 
个 正 数 ， 则 右 移 之 后 在 最 左边 补 n 个 0;， 如 果 数 字 原 先是 负数 ， 则 右 移 之 
后 在 最 左边 补 n 个 1。 下 面 是 对 两 个 8 位 有 符 与 数 作 右 移 的 例子 : 

00001010>>2=00000010 

10001010>>3=11110001 

面试 题 10“ 二 进 制 中 1 的 个 数 ” 就 是 直接 考查 位 运算 的 例子 ， 而 面试 




















题 40“ 数 组 中 只 出 现 一 次 的 数字 ” 面试 题 47“ 不 用 加 减 乘除 做 加 法 ”等 都 
是 根据 位 运算 的 特点 来 解决 问题 。 


面试 题 10: 二 进 制 中 1 的 个 数 


PAA: 请 实现 一 个 函数 ， 输 入 一 个 整数 ， 输 出 该 数 二 进 制 
表示 中 1 的 个 数 。 例 如 把 9 表示 成 二 进 制 是 1001， 有 2 位 是 1。 
此 如 果 输 入 9， 该 函数 输出 2。 


可 能 引起 死人 循环 的 解法 


这 是 一 道 很 基本 的 考查 二 进 制 和 位 运算 的 面试 题 。 题 目 不 是 很 难 ， 
面试 官 提出 问题 之 后 ， 我 们 很 快 就 能 形成 一 个 基本 的 思路 : 先 判 断 整 数 
二 进 制 表示 中 最 右边 一 位 是 不 是 1。 接 着 把 输入 的 整数 右 移 一 位 ， 此 时 
原来 处 于 从 右边 数 起 的 第 二 位 被 移 到 最 右边 了 ， 再 判断 是 不 是 1。 这 样 
每 次 移动 一 位 ， 直 到 整个 整数 变 成 0 为 止 。 现 在 的 问题 变 成 怎么 判断 一 
个 整数 的 最 右边 是 不 是 1 了 。 这 很 简单 ， 只 要 把 整数 和 1 做 位 与 运算 看 结 
果 是 不 是 0 就 知道 了 。1 除 了 最 右边 的 一 位 之 外 所 有 位 都 是 0(。 如 果 一 个 
整数 与 1 做 与 运算 的 结果 是 1， 表 示 该 整数 最 右边 一 位 是 1， 人 否则 是 0。 基 
于 这 个 思路 ， 我 们 很 快 就 能 写 出 如 下 代码 : 
int NumberOfl (int n) 


| 












































int count = 0; 
while (n) 
f 
LRE aeh 
COUDE Efi 
n =n >> 1; 
return count; 





面试 官 看 了 代码 之 后 可 能 会 问 : 把 整数 右 移 一 位 和 把 整数 除 以 2 在 
数学 上 是 等 价 的 ， 那 上 面 的 代码 中 可 以 把 右 移 运 算 换 成 除 以 2 吗 ? BR 
是 否定 的 。 因 为 除法 的 效率 比 移 位 运算 要 低 得 多 ， 在 实际 编程 中 应 尽 可 
能 地 用 移 位 运算 符 代 葵 乘 除法 。 

面试 官 接 下 来 可 能 要 问 的 第 二 个 问题 就 是 : 上 面 的 函数 如 果 输 入 一 








个 负数 ， 比 如 0x80000000， 运 行 的 时 候 会 发 生 什 么 情况 ?把 负数 
0x80000000 右 移 一 位 的 时 候 ， 并 不 是 简单 地 把 最 高 位 的 1 移 到 第 二 位 变 
成 0x40000000， 而 是 0xC0000000。 这 是 因为 移 位 前 是 个 负数 ， 仍 然 要 保 
证 移 位 后 是 个 负数 ， 因 此 移 位 后 的 最 高 位 会 设 为 1。 如 果 一 直 做 右 移 运 
算 ， 最 终 这 个 数字 就 会 变 成 0xYFFFFFFFF 而 陷入 死 循 环 。 


常规 解法 


为 了 避免 死 循环 ， 我 们 可 以 不 右 移 输入 的 数字 i。 首 先 把 i 和 1 做 与 运 
算 ， 判 断 i 的 最 低位 是 不 是 为 1。 接 着 把 1 左 移 一 位 得 到 2， 再 和 i 做 与 运 
算 ， 就 能 判断 i 的 次 低位 是 不 是 1..…. 这 样 反 复 左 移 ， 每 次 都 能 判断 i 的 其 
中 一 位 是 不 是 1。 基 于 这 种 思路 ， 我 们 可 以 把 代码 修改 如 下 : 
int NumberOfl (int n) 
{ 

Int count = 0: 

unsigned int flag = 1; 

while (flag) 




















if(n & flag) 
count ++; 


flag = flag << 1; 


return count; 


这 个 解法 中 循环 的 次 数 等 于 整数 二 进 制 的 位 数 ，32 位 的 整数 需要 循 
环 32 次 。 下 面 再 介绍 一 种 算法 ， 整 数 中 有 几 个 1 就 只 需要 循环 几 次 。 


能 给 面试 官 带 来 惊喜 的 解法 


在 分 析 这 种 算法 之 前 ， 我 们 先 来 分 析 把 一 个 数 减 去 1 的 情况 。 如 果 
一 个 整数 不 等 于 0， 那 么 该 整数 的 二 进 制 表示 中 至 少 有 一 位 是 1。 先 假设 
这 个 数 的 最 右边 一 位 是 1， 那 么 减 去 1 时 ， 最 后 一 位 变 成 0 而 其 他 所 有 位 
都 保持 不 变 。 也 就 是 最 后 一 位 相当 于 做 了 取 反 操作 ， 由 1 变 成 了 0。 

接 下 来 假设 最 后 一 位 不 是 1 而 是 0 的 情况 。 如 果 该 整数 的 二 进 制 表示 
中 最 右边 1 位 于 第 m 位 ， 那 么 减 去 1 时 ， 第 m 位 由 1 变 成 0， 而 第 m 位 之 后 
的 所 有 0 都 变 成 1， 整 数 中 第 m 位 之 前 的 所 有 位 都 保持 不 变 。 举 个 例子 : 
一 个 二 进 制 数 1100， 它 的 第 二 位 是 从 最 右边 数 起 的 一 个 1。 减 去 1 后 ， 第 


} 



































人 
是 1011。 

在 前 面 两 种 情况 中 ， 我 们 发 现 把 一 个 整数 减 去 1， 都 是 把 最 右边 的 1 
变 成 0。 如 果 它 的 右边 还 有 0 的 话 ， 所 有 的 0 都 变 成 1， 而 它 左 边 所 有 位 都 
保持 不 变 。 接 下 来 我 们 把 一 个 整数 和 它 减 去 1 的 结果 做 位 与 运算 ， 相 当 
于 把 它 最 右边 的 1 变 成 0。 还 是 以 前 面 的 1100 为 例 ， 它 减 去 1 的 结果 是 
1011。 我 们 再 把 1100 和 1011 做 位 与 运算 ， 得 到 的 结果 是 1000。 我 们 把 
1100 最 右边 的 1 变 成 了 0， 结 果 刚 好 就 是 1000。 

我 们 把 上 面 的 分 析 总 结 起 来 就 是 : 把 一 个 整数 减 去 1， 再 和 原 整数 
做 与 运算 ， 会 把 该 整数 最 右边 一 个 1 变 成 0。 那 么 一 个 整数 的 二 进 制 表示 
中 有 多 少 个 1， 就 可 以 进行 多 少 次 这 样 的 操作 。 基 于 这 种 思路 ， 我 们 可 
以 写 出 新 的 代码 : 
int NumberOfl (int n) 

{ 


int count = 0; 


while (n) 
++ count; 
n= (nee LT) 1 oy 
return count; 
} 
源 代码 : 


本 题 完整 的 源 代码 详 见 10_NumberOf1lInBinary 项 目 。 

测试 用 例 : 

e IER (包括 边 界 值 1、0x7FFFFFFF) 。 

o 人 负数 (包括 边界 值 0x80000000、0xFFFFFFFF) 。 

e 0. 

本 题 考 点 : 

e 考查 对 二 进 制 及 位 运算 的 理解 。 

e 考查 分 析 、 调 试 代码 的 能 力 。 如 果 应 聘 者 在 面试 过 程 中 采用 的 
是 第 一 种 思路 ， 当 面试 官 提示 他 输入 负数 将 会 出 现 问题 时 ， 面 试 官 会 期 
待 他 能 在 心中 运行 代码 ， 自 己 找 出 运行 出 现 死 循环 的 原因 。 这 要 求 应 聘 


者 有 一 定 的 调试 功底 。 
相关 题目 : 


e 用 一 条 语句 判断 一 个 整数 是 不 是 2 的 整数 次 方 。 一 个 整数 如 果 是 
2 的 整数 次 方 ， 那 么 它 的 二 进 制 表示 中 有 且 只 有 一 位 是 1， 而 其 他 所 有 位 
都 是 0。 根 据 前 面 的 分 析 ， 把 这 个 整数 减 去 1 之 后 再 和 它 自己 做 与 运算 ， 
这 个 整数 中 唯一 的 1 就 会 变 成 0。 

e 输入 两 个 整数 m 和 n， 计 算 需 要 改变 m 的 二 进 制 表 示 中 的 多 少 位 
才能 得 到 n。 比 如 10 的 二 进 制 表示 为 1010，13 的 二 进 制 表示 为 1101， 需 
要 改变 1010 中 的 3 位 才能 得 到 1101。 我 们 可 以 分 为 两 步 解决 这 个 问题 ， 
第 一 步 求 这 两 个 数 的 异 或 ， 第 二 步 统计 异 或 结果 中 1 的 位 数 。 


举一反三 : 


把 一 个 整数 减 去 1 之 后 再 和 原来 的 整数 做 位 与 运算 ， 得 到 的 结果 相当 于 是 把 整数 的 二 进 第 
表示 中 的 最 右边 一 个 1 变 成 0。 很 多 二 进 制 的 问题 都 可 以 用 这 个 思路 解决 。 


25 ”本 章 小 结 


本 章 着 重 介 绍 应 聘 者 在 面试 之 前 应 该 认真 准备 的 基础 知识 。 为 了 应 
Oe A ele 


面试 官 通常 采用 概念 题 、 代 码 分 析 题 及 编程 题 这 3 种 常见 题 型 来 考 
查 应 聘 者 对 某 一 编程 语言 的 掌握 程度 。 本 章 的 2.2 节 讨论 了 C++/C# 语 言 
这 3 种 题 型 的 常见 面试 题 。 

数据 结构 题目 一 直 是 面试 官 考查 的 重点 。 数 组 和 字符 串 是 两 种 最 基 
本 的 数据 结构 。 链 表 应 该 是 面试 题 中 使 用 频率 最 高 的 一 种 数据 结构 。 如 
果 面 试 官 想 加 大 面试 的 难度 ， 他 很 有 可 能 会 选用 与 树 (尤其 是 二 又 树 ) 
相关 的 面试 题 。 由 于 栈 与 递归 调用 密切 相关 ， 队 列 在 图 (包括 树 〉 的 宽 
度 优 先 人 遍历 中 需要 用 到 ， 因 此 应 聘 者 也 需要 掌握 这 两 种 数据 结构 。 

算法 是 面试 官 喜 欢 考 查 的 另外 一 个 重点 。 查 找 〈 特 别 是 二 分 查找 ) 
和 排序 (特别 是 快速 排序 和 归并 排序 〉 是 面试 中 最 经 常 考查 的 算法 ， 应 
聘 者 一 定 要 熟练 掌握 。 另 外 ， 应 聘 者 还 要 掌握 分 析 时 间 复 杂 度 的 方法 ， 
人 
能 大 不 相同 。 

位 运算 是 针对 二 进 制 数字 的 运算 规律 。 只 要 应 聘 者 熟练 掌握 了 二 进 
制 的 与 、 或 、 异 或 运算 及 左 移 、 右 移 操 作 ， 就 能 解决 与 位 运算 相关 的 面 


试题 。 
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Roe ry HY NAS 


3.1 面试 官 谈 代 码 质量 


“一 般 会 考查 代码 的 容错 处 理 能 力 ， 对 一 些 特别 的 输入 会 询问 应 聘 人 员 是 否 考 虑 、 如 何 处 
理 。 不 能 容忍 代码 只 是 针对 一 种 假想 的 ‘正常 值 * 进 行 处 理 ， 不 考虑 异常 状况 ， 也 不 考虑 资源 的 


收 等 问题 。” 
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一 一 笑 焰 《支付 宝 ， 高 级 安全 测试 工程 师 ) 


“如 果 是 因为 粗心 犯错 ， 可 以 原谅 ， 因 为 毕 竞 面试 的 时 候 会 紧张 ;不 能 容忍 的 是 ， 该 掌握 
的 知识 点 却 没有 掌握 ， 而 且 提 醒 了 还 不 知道 。 比 如 下 面 的 : 















































3 VE: 1 由 于 精度 原因 不 能 用 等 号 判断 两 个 小 数 是 否 相等 ， 请 参考 面试 题 11“ 数 值 的 整数 
次 方 ”。 








——4 3] (Autodesk, Software Development Manager) 
“最 不 能 容忍 功能 错误 ， 忽 略 边界 情况 。” 

















一 一 开关 (Intel, Software Engineer) 


“UR REY EAR ae PRB A ABE TC BIE fA SE Ty eR ASB — ie ih 

的 数据 结构 ， 这 会 让 面试 官印 象 大 打折 扣 ， 因 为 这 个 只 能 说 明 他 程序 写 得 太 少 ， 不 够 熟悉 。” 

- 吴斌 (NVidia, Graphics Architect) 

“我 会 从 程序 的 正确 性 和 和 鲁 棒 性 两 方面 检验 代码 的 质量 。 会 关注 对 输入 参数 的 检查 、 处 理 

错误 和 异常 的 方式 、 命 名 方式 等 。 对 于 没有 工作 经 验 的 学 生 ， 程 序 正确 性 之 外 的 错误 基本 都 能 

ee 对 于 有 工作 经 验 的 人 ， 不 能 容忍 考虑 不 周到 、 有 了 明显 
g 鲁 棒 性 错误 。” 






































































































































一 一 田 超 〈 微 软 ，SDE ID 
3.2 ”代码 的 规范 性 


面试 官 是 根据 应 聘 者 写 出 的 代码 来 决定 是 否 录用 他 的 。 如 果 应 聘 者 
代码 写 得 不 够 规范 ， 影 啊 面 试 官 阅读 代码 的 兴致 ， 那 面试 官吏 会 默默 地 








减 去 几 分 。 如 图 3.1 所 示 ， 书 写 、 布 局 和 命名 都 决定 看 代码 的 规范 性 。 


清晰 的 书写 清晰 的 布局 合理 的 命名 





规范 的 代码 


图 3.1 影响 代码 规范 性 的 因素 : 书写 、 布 局 和 命名 


首先 ， 规 范 的 代码 书写 清晰 。 绝 大 部 分 面试 都 是 要 求 应 聘 者 在 白 纸 
或 者 白板 上 书写 。 由 于 现代 人 已 经 习惯 了 毅 键 盘 打 字 ， 手 写 变 得 越 来 越 
不 习惯 ， 因 此 写 出 来 的 字 渡 草 难 辨 。 虽 然 应 聘 者 没有 必要 为 了 面试 特意 
去 练 字 ， 但 在 面试 过 程 中 减 慢 写字 的 速度 ， 尽 量 把 每 个 字母 写 清楚 还 是 
很 有 必要 的 。 不 用 担心 没有 时 间 去 写 代码 ， 通 常 编程 面试 的 代码 量 都 不 
会 超过 50 行 ， 书 写 不 用 花 多 少时 间 ， 关 键 是 在 写 代码 之 前 形成 清晰 的 思 
路 并 能 把 思路 用 编程 语言 清楚 地 书写 出 来 。 

其 次 ， 规 范 的 代码 布局 清晰 。 平 时 程序 员 在 集成 开发 环境 如 Visual 
Studio 里 面 写 代码 ， 依 靠 专业 工具 调整 代码 的 布局 ， 加 入 合理 的 缩 进 并 
让 括号 对 齐 成 对 呈现 。 离 开 了 这 些 工 具 手 写 代 码 ， 我 们 就 要 格外 注意 布 
局 问题 。 当 循环 、 判 断 较 多 ， 逻 辑 较 复 杂 时 ， 缩 进 的 层次 可 能 会 比较 
多 。 如 果 布 局 不 够 清晰 ， 缩 进 也 不 能 体现 代码 的 逻辑 ， 面 试 官 面 对 这 样 
的 代码 将 会 头 坚 脑 胀 。 

最 后 ， 规 范 的 代码 命名 合理 。 很 多 初学 编程 的 人 在 写 代 码 时 总 是 习 
惯用 最 简单 的 名 字 来 俞 名 ， 变 量 名 是 i、j、k， 函 数 名 是 f、g、h。 由 于 
这 样 的 名 字 不 能 告诉 读者 对 应 的 变量 或 者 函数 的 意义 ， 代 码 一 长 就 会 变 
得 星 涩 难 懂 。 强 烈 建 议 应 聘 者 在 写 代码 的 时 候 ， 用 完整 的 英文 单词 组 合 
命名 变量 和 函数 ， 比 如 函数 需要 传 入 一 个 二 又 树 的 根 结 点 作为 参数 ， 则 
可 以 把 该 参数 命名 为 BinaryTreeNode” pRoot， 不 要 因为 这 样 会 多 写 几 个 
字母 而 觉得 抹 烦 。 如 果 一 眼 能 看 出 变量 、 函 数 的 有 用途， 应聘 者 就 能 避 人 锡 
自己 搞 混 消 而 犯 一 些 低级 的 错误 。 同 时 合理 的 命名 也 能 让 面试 官 一 眼 就 
a 
最 小 值 。 


面试 小 提示 : 















































应 聘 者 在 写 代 码 的 时 候 ， 最 好 用 完整 的 英文 单词 组 合 命名 变量 和 函 
数 ， 以 便 面 试 官能 一 眼 读 异 代码 的 意图 。 


3.3 ”代码 的 完整 性 


在 面试 的 过 程 中 ， 面 试 官 会 非常 关注 应 聘 者 考虑 问题 是 否 周 全 。 面 
试 官 通过 检查 代码 是 否 完整 来 考查 应 聘 者 的 思维 是 否 全面 。 通 常 面试 官 
会 检查 应 聘 者 的 代码 是 否 完成 了 基本 功能 、 输 入 边界 值 是 否 能 得 到 正确 
的 输出 、 是 否 对 各 种 不 合 规范 的 非法 输入 做 出 了 合理 的 错误 处 理 。 


1. 从 3 方面 确保 代码 的 完整 性 


应 聘 者 在 写 代码 之 前 ， 首 先 要 把 可 能 的 输入 都 想 清 楚 ， 从 而 避免 在 
程序 中 出 现 各 种 各 样 的 质量 漏洞 。 也 就 是 说 在 编码 之 前 要 考虑 单元 测 
试 。 如 采 能 够 设计 全 面 的 单元 测试 用 例 并 在 代码 中 体现 出 来 ， 那 么 写 出 
的 代码 目 然 也 就 是 完整 正确 的 了 。 通 种 我 们 可 以 从 功能 测试 、 边 界 测试 
和 人 负面 测试 三 方面 设计 测试 用 例 ， 以 确保 代码 的 完整 性 (如 图 3.2 所 
ee 























图 3.2 ”从 功能 测试 、 边 界 测试 、 负 面 测试 3 个 方面 设计 测试 用 例 ， 以 保证 代码 的 完整 性 


首先 要 考虑 的 是 普通 功能 测试 的 测试 用 例 。 我 们 首先 要 保证 写 出 的 
代码 能 够 完成 面试 官 要 求 的 基本 功能 。 比 如 面试 题 要 求 完成 的 功能 是 把 
字符 串 转 换 成 整数 ， 我 们 就 可 以 考虑 输入 字符 串 *123” 来 测试 目 己 写 的 
代码 。 这 里 要 把 零 、 正 数 和 负数 都 考虑 进去 。 

考虑 功能 测试 的 时 候 ， 我 们 要 尽量 突破 常规 思维 的 限制 。 面 试 的 时 
候 我 们 经 党 受到 惯性 思维 的 限制 ， 从 而 看 不 到 更 多 的 功能 需求 。 比 如 面 





试题 12“ 打 印 1 到 最 大 的 n 位 数 ”， 很 多 人 觉得 这 题 很 简单 。 最 大 的 3 位 数 
是 999、 最 大 的 4 位 数 是 9999， 这 些 数字 很 容易 就 能 算出 来 。 但 是 最 大 的 
n 位 数 都 能 用 int 型 表示 吗 ? 超出 int 的 范围 我 们 可 以 考虑 Iong ”long 类 型 ， 
超出 long long 能 够 表示 的 范围 呢 ? 面试 官 是 不 是 要 求 考 虑 任意 大 的 数 
字 ? 如 果 面 试 官 确认 题目 要 求 的 是 任意 大 的 数字 ， 那 么 这 个 题目 就 是 一 
个 大 数 问题 ， 此 时 我 们 需要 特殊 的 数据 结构 来 表示 数字 ， 比 如 用 字符 串 
或 者 数组 来 表示 大 的 数字 ， 以 确保 不 会 溢出 。 

其 次 需要 考虑 各 种 边界 值 的 测试 用 例 。 很 多 时 候 我 们 的 代码 中 都 会 
有 循环 或 者 递归 。 如 果 我 们 的 代码 是 基于 循环 ， 那 么 结束 循环 的 边界 条 
件 是 否 正 确 ? 如 果 是 递归 ， 递 归 终 止 的 边界 值 是 否 正 确 ? 这 些 都 是 边界 
测试 时 要 考虑 的 用 例 。 还 是 以 字符 串 转 换 成 整数 的 问题 为 例 ， 我 们 写 出 
的 代码 应 该 确保 能 够 正确 转换 最 大 的 正 整 数 和 最 小 的 负 整 数 。 

最 后 还 需要 考虑 各 种 可 能 的 错误 的 输入 ， 也 就 是 通常 所 说 的 负面 测 
试 的 测试 用 例 。 我 们 写 出 的 函数 除了 要 顺利 地 完成 要 求 的 功能 之 外 ， 当 
输入 不 符合 要 求 的 时 候 还 能 做 出 合理 的 错误 处 理 。 在 设计 把 字符 串 转 换 
成 整数 的 函数 的 时 候 ， 我 们 就 要 考虑 当 输入 的 字符 串 不 是 一 个 数字 ， 比 
如 ”1a2b3c”， 我 们 怎么 告诉 函数 的 调用 者 这 个 输入 是 非法 的 。 

前 面 我 们 说 的 都 是 要 全 面 考 虑 当前 需求 对 应 的 各 种 可 能 输入 。 在 软 
件 开发 过 程 中 ， 永 远 不 变 的 就 是 需求 会 一 直 改 变 。 如 果 我 们 在 面试 的 时 
候 写 出 的 代码 能 够 把 将 来 需求 可 能 的 变化 都 考虑 进去 ， 在 需求 发 生变 化 
的 时 候 能 够 尽量 减少 代码 改动 的 风险 ， 那 我 们 就 癌 面 试 官 展示 了 上 自己 对 
程序 可 扩展 性 和 可 维护 性 的 理解 ， 通 过 面试 就 是 水 到 渠 成 的 事情 了 。 请 
参考 面试 题 14“ 调 整数 组 顺序 使 奇数 位 于 侦 数 前 面 * 中 关于 可 扩展 性 和 可 
维护 性 的 讨论 。 


2. 3 种 错误 处 理 的 方法 


通常 我 们 有 3 种 方式 把 错误 信息 传递 给 函数 的 调用 者 。 第 一 种 方式 
是 函数 用 返回 值 来 告知 调用 者 是 否 出 错 。 比 如 很 多 Windows 的 API 束 是 
这 个 类 型 。Windows 中 很 多 API 的 返回 值 为 0 表示 API 调 用 成 功 ， 而 返回 
值 不 为 0 表示 在 API 调 用 的 过 程 中 出 错 了 。 微 软 为 不 同 的 非 零 返 回 值 定 义 
了 不 同 的 意义 ， 调 用 者 可 以 根据 这 些 返 回 值 判断 出 错 的 原因 。 这 种 方式 
最 大 的 问题 是 使 用 不 便 ， 因 为 函数 不 能 直接 把 计算 结果 通过 返回 值 赋值 
eer Ree ral a ee rer eae 
函数 。 

第 二 种 方式 是 当 发 生 错 误 时 设置 一 个 全 局 变量 。 此 时 我 们 可 以 在 返 
回 值 中 传递 计算 结果 了 。 这 种 方法 比 第 一 种 方法 使 用 起 来 更 加 方便 ， 因 















































为 调用 者 可 以 直接 把 返回 值 赋值 给 其 他 变量 或 者 作为 参数 传递 给 其 他 函 
数 。Windows 的 很 多 API 运 行 出 错 之 后 ， 也 会 设置 一 个 全 局 变量 。 我 们 
HI OEE H K GetLastError) HAA Ran E KHI E E, 从 而 得 
知 出 错 的 原因 。 但 这 个 方法 有 个 问题 : 调用 者 很 容易 就 会 忘记 去 检查 全 
局 变量 ， 因 此 在 调用 出 错 的 时 候 忘记 做 相应 的 错误 处 理 ， 从 而 留 ITE 
E, 

第 三 种 方式 是 异常 。 当 函数 运行 出 错 的 时 候 ， 我 们 就 抛 出 一 个 异 
常 ， 我 们 还 名 以 根据 不 同 的 出 错 原因 定义 不 同 的 异 常 类 型 。 因 此 函数 的 
调用 者 根据 异常 的 类 型 就 能 知道 出 错 的 原因 ， 从 而 做 相应 的 处 理 。 男 
外 ， 我 们 能 显 式 划分 程序 正常 运行 的 代码 块 (try 模 块 ) 和 处 理 异 常 的 代 
码 块 (catch 模 块 ，， 人 逻辑 比较 清晰 。 异 常 在 高 级 语言 如 C# 中 是 强烈 推 
荐 的 错误 处 理 方式 ， 但 有 些 早 期 的 语言 比如 C 语 言 还 不 支持 异常 。 另 
Ay Fe na ren eure eee eee 

很 大 的 影响 。 

上 述 3 种 错误 处 理 的 方式 各 有 其 优 缺 点 〈 如 表 3.1 所 示 ) 。 那 么 ， 面 
试 的 时 候 我 们 该 采用 哪 种 方式 昵 ? 这 要 看 面试 官 的 需求 。 在 听 到 面试 官 
的 题目 之 后 ， 我 们 要 尽快 分 析出 可 能 存在 哪些 非法 的 输入 ， 并 和 面试 官 
讨论 该 如 何 处 理 这 些 非 法 输入 。 









































表 3.1 返回 值 、 全 局 变量 和 有 异种 三 种 错误 处 理 方 式 的 优 缺 点 比较 


和 系统 API -H 不 能 方便 地 使 用 计算 结果 
全 局 变量 能 够 方便 地 使 用 计算 结果 用 户 可 能 会 忘记 检查 全 局 变量 


可 以 为 不 同 的 出 错 原因 定义 不 同 异常 类 | 有 些 语言 不 支持 异常 ， 抛 出 异常 时 对 性 能 有 负面 
型 ， 逻 辑 清晰 明了 影响 








面试 题 11: 数值 的 整数 次 方 

题目 : 实现 函数 double Power (double base, int 
exponent) ， 求 base 的 exzponent 次 方 。 不 得 使 用 库 函 数 ， 同 时 不 
需要 考虑 大 数 问题 。 

我 们 都 知道 在 C 语 言 的 库 中 有 一 个 pow 函 数 可 以 用 来 求 乘 方 ， 本 题 














要 求实 现 类 似 于 pow 的 功能 。 和 要求 实现 特定 库 函 数 〈 特 别 是 处 理 数 值 和 
FAT BAP BO 的 功能 ， 古 一 类 帝 见 的 面试 题 。 在 本 书 收集 的 面试 题 
中 ， 除 了 这 个 题目 外 ， 还 有 面试 题 49“ 把 字符 串 转换 成 整数 ?要 求实 现 库 
函数 atoi 的 功能 。 这 瓯 要求 我 们 平时 编程 的 时 候 除了 能 熟练 使 用 库 函 数 
外 ， 更 重要 的 是 要 理解 库 函 数 的 实现 原理 。 


目 以 为 题目 简单 的 解法 


由 于 不 需要 考虑 大 数 问 题 ， 这 道 题 看 起 来 很 简单 ， 可 能 不 少 应 聘 者 
在 看 到 题目 30 秒 后 就 能 写 出 如 下 的 代码 : 











double Power(double base, int exponent) 
{ 
double result = 1.0; 
for(int i= 1; i <= exponent; ++i) 
result *= base; 


return result; 


} 





不 过 遗憾 的 是 ， 写 得 快 不 一 定 就 能 得 到 面试 官 的 青睐 ， 因 为 面试 官 
会 问 要 是 输入 的 指数 Cexponent) 小 于 1 即 是 零 和 负数 的 时 候 怎么 办 ? 上 
面 的 代码 完全 没有 考虑 ， 只 包括 了 指数 是 正 数 的 情况 。 


全 面 但 不 够 高 效 的 解法 ， 我 们 离 Offer 已 经 很 近 了 


我 们 知道 当 指数 为 负数 的 时 候 ， 可 以 先 对 指数 求 绝对 值 ， 然 后 算出 
次 方 的 结果 之 后 绸 取 倒数 。 既 然 有 求 倒数 ， 我 们 很 目 然 就 要 想到 有 没有 
可 能 对 0 求 倒 数 ， 如 有 果 对 0 求 倒数 怎么 从? SERB Chase) 是 零 上 且 指 数 是 
负数 的 时 候 ， 如 果 不 做 特殊 处 理 ， 就 会 出 现 对 0 求 倒 数 从 而 导致 程序 运 
行 出 错 。 怎 么 告诉 函数 的 调用 者 出 现 了 这 种 错误 ?前 面 提 到 我 们 可 以 采 
用 3 种 方法 : 返回 值 、 全 局 代码 和 腊 常 。 面 试 的 时 候 可 以 向 面试 官 曾 述 
每 种 方法 的 优 缺 皮 ， 然 后 一 起 讨论 决定 选用 哪 种 方式 。 

最 后 需要 指出 的 是 ， 由 于 0 的 0 次 方 在 数学 上 是 没有 意义 的 ， 因 此 无 
论 是 输出 0 还 是 1 都 是 可 以 接受 的 ， 但 这 都 需要 和 面试 官 说 清楚 ， 表 明 我 
们 已 经 考虑 到 这 个 边界 值 了 。 
ne ee peer 
Zou F: 

















bool g_InvalidInput = false; 


double Power (double base, int exponent) 
{ 
g_InvalidInput = false; 


if(equal(base, 0.0) && exponent < 0) 
{ 

g_InvalidInput = true; 

return 0.0; 


} 


unsigned int absExponent = (unsigned int) (exponent); 
if (exponent < 0) 
absExponent = (unsigned int) (-exponent) ; 


double result = PowerWithUnsignedExponent (base, absExponent) ; 
if(exponent < 0) 
result = 1.0 / result; 


return result; 


} 


double PowerWithUnsignedExponent (double base, unsigned int exponent) 
{ 
double result = 1.0; 
for(int i = 1; i <= exponent; ++i) 
result *= base; 


return result; 


} 


bool equal (double numl, double num2) 
{ 
if(({numl - num2 > -0.0000001) 
&& (numl -~ num2 < 0.0000001)) 
return true; 
else 
return false; 








在 上 述 代码 中 我 们 采用 全 局 变量 来 标识 是 否 出 错 。 如 果 出 错 了 ， 则 
返回 的 值 是 (。 但 为 了 区 分 是 出 错 的 时 候 返 回 的 0， 还 是 确 数 为 0 的 时 候 








正常 运行 返回 的 0， 我 们 还 定义 了 一 个 全 局 变量 g_InvalidInput。 当 出 错 
时 ， 这 个 变量 被 设 为 tue， 否 则 为 false。 这 样 做 的 好 处 是 ， 我 们 可 以 把 
返回 值 直接 传递 给 其 他 变量 ， 比 如 写 double result=Power (2,3) ， 也 可 
以 把 函数 的 返回 值 直 接 传递 给 其 他 需要 double 型 参数 的 函数 。 但 缺点 是 
这 个 函数 的 调用 者 有 可 能 会 态 记 去 检查 g_InvalidInput 以 判断 是 否 出 错 ， 
留 下 了 安全 隐患 。 由 于 有 优点 也 有 缺点 ， 因 此 我 们 在 写 代 码 之 前 要 和 面 
试 官 讨论 采用 哪 种 出 错 处 理 方式 最 合适 。 

一 个 细节 值得 我 们 注意 : 在 判断 底数 base 是 不 是 等 于 0 时 ， 不 能 直 
接 写 base==0， 这 是 因为 在 计算 机 内 表示 小 数 时 (包括 float 和 double 型 小 
数 ) 都 有 误差 。 判 断 两 个 小 数 是 否 相 等 ， 只 能 判断 它们 之 又 的 绝对 值 是 
不 是 在 一 个 很 小 的 范围 内 。 如 果 两 个 数 相差 很 小 ， 束 可 以 认为 它们 相 
等 。 这 就 是 我 们 定义 函数 equal 的 原因 。 

面试 小 提示 : 

由 于 计算 机 表示 小 数 〈 包 括 foat 和 double 型 小 数 ) 都 有 误差 .我们 不 能 直接 用 等 号 (==) 
PISPA. MRP AMERA Hei Tooo, MAUNE 

此 时 我 们 考虑 得 已 经 很 周详 了 ， 已 经 能 够 达到 很 多 面试 官 的 要 求 
了 。 但 是 如 果 我 们 碰 到 的 面试 官 是 一 个 在 效率 上 追求 完美 的 人 ， 那 么 他 
有 可 能 会 提醒 我 们 函数 PowerWithUnsignedExponent 还 有 更 快 的 办 法 。 


全 面 又 高 效 的 解法 ， 确 保 我 们 能 拿 到 Offer 


如 果 输 入 的 指数 exponent 为 32， 我 们 在 函数 
PowerWithUnsignedExponent 的 循环 中 需要 做 31 次 乘法 。 但 我 们 可 以 换 
一 种 思路 考虑 : 我 们 的 目标 是 求 出 一 个 数字 的 32 次 方 ， 如 果 我 们 已 经 知 
道 了 它 的 16 次 方 ， 那 么 只 要 在 16 次 方 的 基础 上 再 平方 一 次 就 可 以 了 。 而 
16 次 方 是 8 次 方 的 平方 。 这 样 以 此 类 推 ， 我 们 求 32 次 方 只 需要 做 5 次 乘 
法 : 先 求 平 方 ， 在 平方 的 基础 上 求 4 次 方 ， 在 4 次 方 的 基础 上 求 8 次 方 ， 
在 8 次 方 的 基础 上 求 16 次 方 ， 最 后 在 16 次 方 的 基础 上 求 32 次 方 。 

也 就 是 说 ， 我 们 可 以 用 如 下 公式 求 a 的 n 次 方 : 

n lal! n 为 偶数 

a = hgga n 为 奇数 

这 个 公式 看 起 来 是 不 是 眼熟 ?我 们 前 面 在 介绍 用 O dogn) 时间 求 
斐 波 那 契 数列 时 ， 就 讨论 过 这 个 公式 ， 这 个 公式 很 容易 就 能 用 递归 来 实 
现 。 新 的 PowerWithUnsignedExponent 代 码 如 下 : 
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double PowerWithUnsignedExponent (double base, unsigned int exponent) 


{ 


if (exponent == 0) 
return 1; 
if (exponent == 1) 


return base; 


double result = PowerWithUnsignedExponent (base, exponent >> 1); 
result *= result; 
if (exponent & 0x1 == 1) 


result *= base; 
return result; 


) 

最 后 再 提醒 一 个 细节 : 我 们 用 右 移 运算 符 代 蔡 了 除 以 2?， 用 位 与 运 
算 符 代替 了 求 余 运算 符 〈%) 来 判断 一 个 数 是 奇数 还 是 偶数 。 位 运算 的 
效率 比 乘除 法 及 求 余 运算 的 效率 要 高 很 多 。 既 然 要 优化 代码 ， 我 们 就 把 
优化 做 到 极致 。 

在 面试 的 时 候 ， 我 们 可 以 主动 提醒 面试 官 注意 代码 中 的 两 处 细 市 
(判断 base 是 否 等 于 0 和 用 位 运算 丛 代 乘除 法 及 求 余 运算 ) ， 让 他 知道 
我 们 对 编程 的 细 布 很 重视 。 细 节 很 重要 ， 因 为 细节 决定 成 败 ， 一 两 个 好 
的 细 市 说 不 定 就 能 让 面试 官 下 定 决心 给 我 们 Offer。 


源 代码 : 

本 题 完 整 的 源 代码 详 见 11_Power 项 目 。 
测试 用 例 : 

把 底数 和 指数 分 别 设 为 正 数 、 负 数 和 和 零 。 
本 题 考点 : 


e 考 售 思 维 的 全 面 性 。 这 个 问题 本 里 不 难 ， 但 能 顺利 通过 的 应 聘 
者 不是 很 多 。 有 很 多 人 会 忽视 乓 数 为 0 而 指数 为 负数 时 的 错误 处 理 。 
pr 对 效率 要 求 比较 高 的 面试 官 还 会 考查 应 聘 者 快速 做 乘 方 的 能 








面试 题 12: 打印 1 到 最 大 的 mn 位 数 
题目 : 输入 数字 n， 按 顺序 打印 出 从 1 最 大 的 n 位 十 进 制 


数 。 比 如 输入 3， 则 打印 出 1、2、3 一 直到 最 大 的 3 位 数 即 999。 
跳 进 面试 官 陷阱 

这 个 题目 看 起 来 很 简单 。 我 们 看 到 这 个 问题 之 后 ， 最 容易 想到 的 办 
法 是 先 求 出 最 大 的 n 位 数 ， 然 后 用 一 个 循环 从 1 开始 逐个 打印 。 于 是 我 们 
很 容易 就 能 写 出 如 下 的 代码 : 


void PrintlToMaxOfNDigits_1l(int n) 


{ 





int number = 1; 

int i = 0; 

while (i++ < n) 
number *= 10; 


for(i = I; i <<numbers +42) 
Dei >) PONE ss Be 


初 看 之 下 好 像 没有 问题 ， 但 如 果 仔 细 分 析 这 个 问题 ， 我 们 就 能 注意 
到 面试 官 没有 规定 n 的 范围 。 当 输入 的 n 很 大 的 时 候 ， 我 们 求 最 大 的 n 位 
数 是 不 是 用 整 型 (int) 或 者 长 整 型 (long long) 都 会 溢出 ”也 就 是 说 我 
们 需要 考虑 大 数 问题 。 这 是 面试 官 在 这 道 题 里 设置 的 一 个 大 陷阱 。 


在 字符 串 上 模拟 数字 加 法 的 解法 ， 绕 过 陷阱 才能 拿 到 Offer 


经 过 前 面 的 分 析 ， 我 们 很 自然 地 想到 解决 这 个 问题 需要 表达 一 个 大 
数 。 最 肖 用 也 是 最 容易 的 方法 是 用 字符 串 或 者 数组 表达 大 数 。 接 下 来 我 
们 用 字符 串 来 解决 大 数 问题 。 

用 字符 串 表示 数字 的 时 候 ， 最 直观 的 方法 就 是 字符 串 里 每 个 字符 都 
是 '0’ 到 ”9 之 间 的 茶 一 个 字符 ， 用 来 表示 数字 中 的 一 位 。 因 为 数字 最 大 
是 n 位 的 ， 因 此 我 们 需要 一 个 长 度 为 n 十 1 的 字符 串 〈 字 符 串 中 最 后 一 个 
是 结束 符号 \0’;) 。 当 实际 数字 不 够 nm 位 的 时 候 ， 在 字符 串 的 前 半 部 分 补 
0。 


首先 我 们 把 字符 串 中 的 每 一 个 数字 都 初始 化 为 '0'， 然 后 每 一 次 为 字 
符 串 表示 的 数字 加 1， 再 打印 出 来 。 因 此 我 们 只 需要 做 两 件 事 : 一 是 在 
字符 串 表 达 的 数字 上 模拟 加 法 ， 二 是 把 字符 串 表 达 的 数字 打印 出 来 。 
基于 上 面 的 分 析 ， 我 们 可 以 写 出 如 下 代码 : 


} 

















void PrintlToMaxOfNDigits (int n) 
{ 


if(n <= 0) 

return; 
char *number = new char[n + 1]; 
memset (number, 'O', n); 
number[n] = '\0'; 


while (!Increment (number) ) 


PrintNumber (number); 


delete []number; 


在 上 面 的 代码 中 ， 函 数 mcrement 实 现在 表示 数字 的 字符 串 number 上 
增加 1， 而 函数 PrintNumber 打 印 出 number。 这 两 个 看 似 简 单 的 函数 都 瞳 
藏 着 小 小 的 辫 机 。 

我 们 需要 知道 什么 时 候 停 止 在 number 上 增加 1， 即 什么 时 候 到 了 最 
大 的 n 位 数 “999...99”(n 个 9) 。 一 个 最 简单 的 办 法 是 每 次 递增 之 后 ， 都 
调用 库 函 数 strcmp 比 较 表 示 数 字 的 字符 串 number 和 最 大 的 n 位 数 “999... 
99”， 如 果 相 等 则 表示 已 经 到 了 最 大 的 n 位 数 并 终止 递增 。 虽 然 调 用 
stremp 很 简单 ， 但 对 于 长 度 为 n 的 字符 串 ， 它 的 时 间 复 杂 上 度 为 O(n) 。 

我 们 注意 到 只 有 对 “999...99” 加 1 的 时 候 ， 才 会 在 第 一 个 字符 (下 标 
AO) 的 基础 上 产生 进位 ， 而 其 他 所 有 情况 都 不 会 在 第 一 个 字符 上 产生 
进位 。 因 此 当 我 们 发 现在 加 1 时 第 一 个 字符 产生 了 进位 ， 则 已 经 是 最 大 
的 n 位 数 ， 此 时 Increment 返 回 true， 因 此 函数 Print1ToMaxOfNDigits 中 的 
while 循 环 终止 。 如 何在 每 一 次 增加 1 之 后 快速 判断 是 不 是 到 了 最 大 的 n 
位 数 是 本 题 的 一 个 小 陷阱 。 下 面 是 mcrement 函 数 的 参考 代码 ， 它 实现 了 
FAO (1) 时 间 判 断 是 不 是 已 经 到 了 最 大 的 n 位 数 : 


} 


























bool Increment (char* number) 


{ 

bool isOverflow = false; 

int nTakeOver = 0; 

int nLength = strlen (number); 

for(int i = nLength - 1; i >= 0; i --) 

{ 
int nSum = number[i] - '0' + nTakeOver; 
if(i == nLength - 1) 


nSum ++; 


{ 
EEGI = 0) 
isOverflow = true 
else 
{ 
nsum -= 1í 
nTakeOver = 1 
number[i] 三 0" + nSum; 
} 
} 
{ 
number[i] = '0' + nSum; 
break; 
} 


} 


return isOverflow; 
} 

接 下 来 我 们 再 考虑 如 何 打 印 用 字符 串 表 示 的 数字 。 虽 然 库 函数 
Printf 可 以 很 方便 就 能 打印 一 个 字符 串 ， 但 在 本 题 中 调用 printf 并 不 是 最 
合适 的 解决 方案 。 前 面 我 们 提 到 ， 当 数字 不 够 mn 位 的 时 候 ， 我 们 在 数字 
的 前 面 补 0， 打 印 的 时 候 这 些 补 位 的 0 不 应 该 打印 出 来 。 比 如 输入 3 的 时 
候 ， 数 字 98 用 字符 串 表 示 成 “098”。 如 果 直 接 打印 出 098， 就 不 符合 我 们 
的 习惯 。 为 此 我 们 定义 了 函数 PrintNumber， 在 这 个 函数 里 ， 我 们 只 有 
在 倍 到 第 一 个 非 0 的 字符 之 后 才 开 始 打印 ， 直 至 字符 串 的 结尾 。 能 不 能 
ee 是 面试 官 设置 的 另外 一 个 小 陷阱 。 实 现 

ASO F: 








void PrintNumber(char* number) 


bool isBeginningO = true; 
int nLength = strlen (number); 
for(int i = 0; i < nLength; ++ 1) 
{ 
if(isBeginningO && number[i] != '0') 
isBeginningO = false; 
if (!isBeginning0O) 
{ 
printf("%c", number [i]); 
} 
} 
print£("\t"); 


把 问题 转换 成 数字 排列 的 解法 ， 递 归 让 代码 更 简 清 


上 述 思路 虽然 比较 直观 ， 但 由 于 模拟 了 整数 的 加 法 ， 代 码 有 点 长 。 
要 在 面试 短 短 几 十 分 钟 时 间 里 完整 正确 地 写 出 这 么 长 的 代码 ， 对 很 多 应 
聘 者 而 言 不 是 一 件 容易 的 事情 。 接 下 来 我 们 换 一 种 思路 来 考虑 这 个 问 
题 。 如 采 我 们 在 数字 前 面 补 0 的 话 ， 就 会 发 现 n 位 所 有 十 进 制 数 其 实 就 是 
n 个 从 0 到 9 的 全 排列 。 也 就 是 说， 我 们 把 数字 的 每 一 位 都 从 0 到 9 排列 一 
近 ， 就 得 到 了 所 有 的 十 进 制 数 。 只 是 我 们 在 打印 的 时 候 ， 数 字 排 在 前 面 
的 0 我 们 不 打印 出 来 罢了 。 

全 排列 用 递归 很 容易 表达 ， 数 字 的 每 一 位 都 可 能 是 0 一 9 中 的 一 个 
数 ， 然 后 设置 下 一 位 。 弟 归结 束 的 条 件 是 我 们 已 经 设置 了 数 子 的 最 后 一 


位 。 











void PrintlToMaxOfNDigits (int n) 


{ 


} 


if(n <= 0) 
return; 


char* number = new char[n + 1]; 
number[n] = '\0'; 


for (int.i = Of i < 103 TFH 

{ 
number[{0] = i+ '0'; 
PrintlToMaxOfNDigitsRecursively(number, n, 0); 


} 


delete[] number; 


void Print1lToMaxOfNDigitsRecursively(char* number, int length, 
index) 


{ 


} 


了 。 


if (index == length - 1) 
{ 
PrintNumber (number) ; 
return; 


} 


for(int i = 0; i < 10; ++i) 
{ 


number[index + 1] =i + '0'; 


int 


Print1ToMaxOfNDigitsRecursively (number, length, index + 1); 


} 
函数 PrintNumber 和 前 面 第 二 种 思路 中 的 一 样 ， 这 里 就 不 再 重复 


源 代码 : 
本 题 完整 的 源 代 码 详 见 12_Print1ToMaxOfNDigits 项 目 。 
测试 用 例 : 


e 功能 测试 〈 输 入 1、2、3..….. Dies 
e 特殊 输入 测试 (输入 -1、0) 。 


本 题 考点 : 


© 考 奋 解决 大 数 问题 的 能 力 。 面 试 官 出 这 个 题目 的 时 候 ， 他 期 户 
应 聘 者 能 意识 到 这 是 一 个 大 数 问题 ， 同 时 还 期 竺 应 聘 者 能 定义 合适 的 数 
据 表 示 方 式 来 解决 大 数 问题 。 

e 如果 应 聘 者 采用 第 一 种 思路 即 在 数 上 加 1 逐个 打印 的 思路 ， 面 试 
官 会 关注 他 判断 是 否 已 经 到 了 最 大 的 n 位 数 时 采用 的 方法 。 应 聘 者 要 注 
意 到 不 同方 法 的 时 间 效 率 相 差 很 大 。 

e 如果 应 阵 者 采用 第 二 种 思路 ， 面 试 官 还 将 考查 他 用 递归 方法 解 
决 问题 的 能 

e 面试 官 还 将 关注 应 聘 者 打印 数字 时 会 不 会 打印 出 位 于 数字 最 前 
面 的 0。 这 里 能 体现 出 应 聘 者 在 设计 开发 软件 时 是 不 是 会 考虑 用 户 的 使 
用 习惯 。 通 利 我 们 的 软件 设计 和 开发 需要 符合 大 部 分 用 户 的 人 机 交互 习 


惯 。 














本 题 扩展 : 


在 前 面 的 代码 中 ， 我 们 都 是 用 一 个 char 型 字符 表示 十 进 制 数字 的 一 
位 。8 个 bit 的 char 型 字符 最 多 能 表示 256 个 字符 ， 而 十 进 制 数字 只 有 0 一 
的 10 个 数字 。 因 此 用 char 型 字符 囊 来 表示 十 进 制 的 数字 并 没有 充分 利用 
内 存 ， 有 一 些 浪费 。 有 没有 更 高 效 的 方式 来 表示 大 数 ? 

相关 题目 : 

定义 一 个 函数 ， 在 该 函数 中 可 以 实现 任意 两 个 整数 的 加 法 。 由 于 没 
有 限定 输入 两 个 数 的 大 小 范围 ， 我 们 也 要 把 它 当 做 大 数 问题 来 处 理 。 在 
前 面 的 代码 的 第 一 个 思路 中 ， 实 现 了 在 字符 串 表示 的 数字 上 加 1 的 功 
能 ， 我 们 可 以 参考 这 个 思路 实现 两 个 数字 的 相 加 功能 。 另 外 还 有 一 个 需 
要 注意 的 问题 ， 如 果 输 入 的 数字 中 有 负数 ， 我 们 应 该 怎么 去 处 理 ? 

面试 小 提示 ; 

如 果 面 试题 是 关于 n 位 的 整数 并 且 没有 限定 n 的 取 值 范围 ， 或 者 是 输入 任意 大 小 的 整数 ， 
那么 这 个 题目 很 有 可 能 是 需要 考虑 大 数 问题 的 。 字 符 串 是 一 个 简单 、 有 效 的 表示 大 数 的 方法 。 


面试 题 13: 在 O (1) 时 间 删 除 链表 结 点 

































































题目 : 给 定单 同 链 表 的 头 指针 和 一 个 结 点 指针， 定义 一 个 
KAEO C1) 时 间 删 除 该 结 点 。 链 表 结 点 与 函数 的 定义 如 
ike 


struct ListNode 

{ 
int m_nValue; 
ListNode* m_pNext; 


}; 


void DeleteNode (ListNode** pListHead, ListNode* pToBeDeleted) ; 


EA BERKEMBAR BCE PO I IS TC Ba ce ERE SRG RA 
开始 ， 顺 序 过 历 碍 找 要 删除 的 结 点 ， 并 在 链表 中 删除 该 结 点 。 


比如 在 图 3.3 Ca) 所 示 的 链表 中 ， 我 们 想 删除 结 点 1， 可 以 从 链表 的 
头 结 点 a 开始 顺序 过 历 ， 发 现 结 点 h 的 m_pNext 指 向 要 删除 的 结 点 !， 于 古 
我 们 可 以 把 结 皮 h 的 m_pNext 指 向 i 的 下 一 个 结 点 即 结 点 )]。 指 针 调 整 之 
后 ， 我 们 就 可 以 安全 地 删除 结 点 并 保证 链表 没有 断 开 (如 图 3.3 Cb) 所 
示 ) 。 这 种 思路 由 于 需要 顺序 查找 ， 时 间 复 杂 度 目 然 就 是 O Cn) To 


| 
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图 3.3 ”在 链表 中 删除 一 个 结 点 的 两 种 方法 


YE: (Ca) 一 个 链表 。 (b) 删除 结 点 ij 之 前 ， 先 从 链表 的 头 结 点 开始 所 有 历 到 i 前 面 的 一 个 结 
点 h， 把 h 的 m_pNext 指 同 i 的 下 一 个 结 点 j， 再 删除 结 点 i。 (O HAMA A SHA mA, Be 


下 来 再 把 结 点 ij 的 m_pNext 指 向 j 的 下 一 个 结 点 之 后 删除 结 点 j。 这 种 方法 不 用 遍历 链表 上 结 点 ji 前 
面 的 结 点 。 


之 所 以 需要 从 头 开 始 碍 找 ， 是 因为 我 们 需要 得 到 将 被 删除 的 结 点 的 














































































































前 面 一 个 结 点 。 在 单 癌 链表 中 ， 结 点 中 没有 指 同 前 一 个 结 点 的 指针 ， 上 所 
以 只 好 从 链表 的 头 结 点 开始 顺序 查找 。 l 
那 是 不 是 一 定 需要 得 到 被 删除 的 结 点 的 前 一 个 结 点 呢 ? SRE aE 


的 。 我 们 可 以 很 方便 地 得 到 要 删除 的 结 点 的 一 下 结 点 。 如 果 我 们 把 下 一 
个 结 点 的 内 容 复制 到 需要 删除 的 结 点 上 获 兰 原 有 的 内 容 ， 再 把 下 一 个 结 
点 删除 ， 那 是 不 是 就 相当 于 把 当前 需要 删除 的 结 点 删除 了 ? 

我 们 还 是 以 前 面 的 例子 来 分 析 这 种 思路 。 我 们 要 删除 结 点 i， 先 把 i 
的 下 一 个 结 反 j 的 内 容 复 制 到 i， 然 后 把 的 指针 指向 结 反 j 的 下 一 个 结 点 。 





此 时 再 删除 结 点 j)， 其 效果 刚好 是 把 结 点 i 给 删除 了 【如 图 3.3 (c) 所 
AN)? a 

上 述 思 路 还 有 一 个 问题 : 如 果 要 删除 的 结 点 位 于 链表 的 尾部 ， 那 么 
它 束 没有 下 一 个 结 点 ， 怎 么 办 ? 我 们 仍然 从 链表 的 头 结 点 开始 ， 顺 序 遍 
历 得 到 该 结 点 的 前 序 结 点 ， 并 完成 删除 操作 。 

最 后 需要 注意 的 是 ， 如 果 链 表 中 只 有 一 个 结 点 ， 而 我 们 又 要 删除 链 
表 的 头 结 点 (也 是 尾 结 点 ) ， 此 时 我 们 在 删除 结 点 之 后 ， 还 需要 把 链表 
的 头 结 点 设置 为 NULL 。 
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void DeleteNode (ListNode** pListHead, ListNode* pToBeDeleted) 
{ 


1f(!pListHead || !pToBeDeleted) 
return; 


// 要 删除 的 结 点 不 是 尾 结 点 

if (pToBeDeleted->m pNext != NULL) 

{ 
ListNode* pNext = pToBeDeleted->m_pNext; 
pToBeDeleted->m nValue = pNext->m nValue; 
pToBeDeleted->m_pNext = pNext->m_pNext; 


delete pNext; 
pNext = NULL; 


} 
// 链表 只 有 一 个 结 点 ， 删 除 头 结 点 (也 是 尾 结 点 ) 
else if(*pListHead == pToBeDeleted) 


表 中 有 多 个 结 点 ， 删 除 尾 结 点 


> 
© ir 


ListNode* pNode = *pListHead; 
while (pNode->m_pNext != pToBeDeleted) 


pNode = pNode->m_pNext; 
} 


pNode->m_pNext = NULL; 
delete pToBeDeleted; 
pToBeDeleted = NULL; 

} 


接 下 来 我 们 分 析 这 种 思路 的 时 间 复 杂 度 。 对 于 n 一 1 个 非 尾 结 点 而 
言 ， 我 们 可 以 在 O (1) 时 把 下 一 个 结 点 的 内 存 复制 覆盖 要 删除 的 结 
点 ， 并 删除 下 一 个 结 点 ; 对 于 尾 结 点 而 言 ， 由 于 仍然 需要 顺序 查找 ， 时 
间 复 杂 度 是 O (Cn) 。 因 此 总 的 平均 时 间 复 杂 度 是 [ (n 一 1) 'O (1) 十 
O (n) Jm， 结 果 还 是 O (1) ， 符 合 面 试 官 的 要 求 。 

值得 注意 的 是 ， 上 述 代 码 仍然 不 是 完美 的 代码 ， 因 为 它 基 于 一 个 假 


} 








设 : 要 删除 的 结 点 的 确 在 链表 中 。 我 们 需要 O Cn) 的 时 间 才 能 判断 链 
表 中 是 否 包含 某 一 结 点 。 受 到 O (1) 时 间 的 限制 ， 我 们 不 得 不 把 确保 
结 点 在 链表 中 的 责任 推 给 了 函数 DeleteNode 的 调用 者 。 在 面试 的 时 候 ， 
E alee 
常 全 面 。 








源 代码 : 
本 题 完整 的 源 代码 详 见 13_DeleteNodelInList 项 目 。 
测试 用 例 : 


e 功能 测试 《从 有 多 个 结 点 的 链表 的 中 间 删 除 一 个 结 点 ， 从 有 多 
个 结 点 的 链表 中 删除 头 结 点 ， 从 有 多 个 结 点 的 链表 中 删除 尾 结 点 ， 从 只 
有 一 个 结 点 的 链表 中 删除 唯一 的 结 点 ) 。 

e 特殊 输入 测试 〈 指 向 链表 头 结 点 指针 的 为 NULL 指 针 ， 指 向 要 
删除 的 结 点 为 NULL 指 针 ) 。 


本 题 考点 : 


© 考 俘 应 聘 者 对 链表 的 编程 能 力 。 

o 考查 应 聘 者 的 创新 思维 能 力 。 这 道 题 要 求 应 聘 者 打破 常规 的 思 
维 模 式 。 当 我 们 想 删 除 一 个 结 皮 时， 并 不 一 定 要 删除 这 个 结 点 本 里 。 可 
以 先 把 下 一 个 结 点 的 内 容 复 制 出 来 履 六 被 删除 结 点 的 内 容 ， 然 后 把 下 一 
个 结 扣 删除 。 这 种 思路 不 是 很 容易 想到 的 。 

o 考 合 应 聘 者 思维 的 全 面 性 。 即 使 应 聘 者 想到 删除 下 一 个 结 点 这 
个 办 法 ， 也 未 必 能 通过 这 轮 面 试 。 应 聘 者 要 全 面 考虑 到 删除 的 结 点 位 于 
链表 的 尾部 及 输入 的 链表 只 有 一 个 结 点 这 些 特殊 情况 。 


面试 题 14: 调整 数组 顺序 使 奇数 位 于 偶数 前 面 


题目 : 输入 一 个 整数 数组 ， 实 现 一 个 函数 来 调整 该 数组 中 
数字 的 顺序 ， 使 得 所 有 奇数 位 于 数组 的 前 半 部 分 ， 所 有 偶数 位 
于 数组 的 后 半 部 分 。 

如 果 不 考 虑 时 间 复 杂 度 ， 了 节 简 单 的 思路 应 该 是 从 头 扫描 这 个 数组 ， 
每 磁 到 一 个 偶数 时 ， 拿 出 这 个 数字 ， 并 把 位 于 这 个 数字 后 面 的 所 有 数字 
往 前 挪动 一 位 。 挪 完 之 后 在 数组 的 末尾 有 一 个 空位 ， 这 时 把 该 侦 数 放 入 
这 个 空位 。 由 于 每 碰 到 一 个 偶数 吕 需 要 移动 O Cn) 个 数字 ， 因 此 总 的 
时 间 复 杂 度 是 O Co) 。 但 是 ， 这 种 方法 不 能 让 面试 官 满意 。 不 过 如 果 








我 们 在 听 到 题目 之 后 号 上 能 说 出 这 个 解法 ， 面 试 官 至 少 会 觉得 我 们 的 思 
维 非常 敏捷 。 


只 完成 基本 功能 的 解法 ， 仅 适用 于 初级 程序 员 


这 个 题目 要 求 把 奇数 放 在 数组 的 前 半 部 分 ， 偶 数 放 在 数组 的 后 半 部 
分 ， 因 此 所 有 的 奇数 应 该 位 于 偶数 的 前 面 。 也 束 是 说 我 们 在 扫描 这 个 数 
组 的 时 候 ， 如 果 发 现 有 偶数 出 现在 奇数 的 前 面 ， 我 们 可 以 交换 它们 的 顺 
序 ， 交 换 之 后 就 符合 要 求 了 。 

因此 我 们 可 以 维护 两 个 指针 ， 第 一 个 指针 初始 化 时 指 癌 数组 的 第 一 
个 数字 ， 它 只 回 后 移动 ; 第 二 个 指针 初始 化 时 指 同 数组 的 最 后 一 个 数 
字 ， 它 只 同 前 移动 。 在 两 个 指针 相遇 之 前 ， 第 一 个 指针 总 是 位 于 第 二 个 
指针 的 前 面 。 如 果 第 一 个 指针 指 同 的 数字 是 俩 数 ， 并 有 旦 第 二 个 指针 指 癌 
的 数字 是 奇数 ， 我 们 就 交换 这 两 个 数字 。 

下 面 以 一 个 具体 的 例子 比如 输入 数组 {1,2,3,4,5} 来 分 析 这 种 思路 。 
在 初始 化 时 ， 把 第 一 个 指针 指向 数组 第 一 个 数字 1， 而 把 第 二 个 指针 指 
癌 最 后 一 个 数字 5 如 图 3.4 (a) BAN) 。 第 一 个 指针 指 问 的 数字 1 是 一 
个 奇数 ， 不 需要 人 处理， 我 们 把 第 一 个 指针 问 后 移动 ， 直 到 人 磅 到 一 个 侦 数 
2。 此 时 第 二 个 指针 已 经 指 癌 了 奇数 ， 因 此 不 需要 移动 。 此 时 两 个 指针 
指 癌 的 位 置 如 图 3.4 Cb) 所 示 。 这 个 时 候 我 们 发 现 偶 数 2 位 于 奇数 5 的 前 
面 ， 符 合 我 们 的 交换 条 件 ， 于 是 交换 两 个 指针 指 同 的 数字 (如 图 
3.4 (c) 所 示 ) 。 

接 下 来 我 们 继续 向 后 移 第 一 个 指针 ， 直 到 碰 到 下 一 个 偶数 4， 并 癌 
前 移动 第 二 个 指针 ， 直 到 磁 到 第 一 个 奇数 3( 如 图 3.4 Cd) 所 示 ) 。 我 
们 发 现 第 二 个 指针 已 经 在 第 一 个 指针 的 前 面 了 ， 表 示 所 有 的 奇数 都 已 经 
在 偶数 的 前 面 了 。 此 时 的 数组 是 {1,5,3,4,2}， 的 确 是 奇数 位 于 数组 的 前 
半 部 分 而 偶数 位 于 后 半 部 分 。 
































图 3.4 调整 数组 {12,3,4,5} 使 得 奇数 位 于 偶数 前 面 的 过 程 


TE: (a) 把 第 一 个 指针 指向 数组 的 第 一 个 数字 ， 第 二 个 指针 指向 最 后 一 个 数字 。 (b) 
向 后 移动 第 一 个 指针 直至 它 指 向 偶数 >， 此 时 第 二 个 指针 指向 奇数 5， 不 需要 移动 。(c) 交换 两 
个 指针 指向 的 数字 。 (D 向 后 移动 第 一 个 指针 直至 它 指 向 偶数 4， 向 前 移动 第 二 个 指针 直至 它 
指向 奇数 3。 由 于 第 二 个 指针 移 到 了 第 一 个 指针 的 前 面 ， 表 明 所 有 的 奇数 都 位 于 偶数 的 前 面 。 


基于 这 个 分 析 ， 我 们 可 以 写 出 如 下 代码 : 


void ReorderOddEven(int *pData, unsigned int length) 















































ata == NULL | length == 0) 


int *pBegin = pData; 


pD 
int *pEnd = pDat length - 1; 


while (pBegin < pEnd) 


{ 
// 向 后 移动 pBegin， 直 到 它 指 向 偶数 
while(pBegin < pEnd && (*pBegin & 0x1) != 0) 
pBegin ++; 
// 向 前 移动 bEnd， 直 到 它 指向 奇 
while(pBegin < pEnd && (* pind & 0x1) == 0) 
pEnd --; 
if(pBegin < pEnd) 
{ 
int temp = *pBegin; 
kpBegin = *pEnd 
kpEnd = temp; 
} 
} 


考虑 可 扩展 性 的 解法 ， 能 秒杀 Offer 


如 果 是 面试 应 届 毕 业 生 或 者 工作 时 间 不 长 的 程序 员 ， 面 试 官 会 满意 
前 面 的 代码 。 但 如 果 应 聘 者 申请 的 是 资深 的 开发 职位 ， 那 面试 官 可 能 会 
接着 问 几 个 问题 。 

面试 官 ， 如果 把 题目 改 成 把 数组 中 的 数 按照 大 小 分 为 两 部 分 ， 所 有 负数 都 在 非 负数 的 前 
面 ， 该 怎么 做 ? 




















应 聘 者 : 这 很 简单 ， 可 以 重新 定义 一 个 函数 。 在 新 的 函数 里 ， 只 要 修改 第 二 个 和 第 三 个 
while 循 环 中 的 判断 条 件 就 行 了 。 

面试 官 ， 如 果 再 把 题目 改 改 ， 变 成 把 数组 中 的 数 分 为 两 部 分 ， 能 被 3 整除 的 数 都 在 不 能 被 
3 整除 的 数 的 前 面 。 怎 么 办 ? 

应 聘 者 : 我 们 还 是 可 以 定义 一 个 新 的 函数 。 在 这 个 函数 中 .……… 

面试 官 : 〈 打 断 应 聘 者 的 话 ) 难道 就 没有 更 好 的 办 法 ? 

这 个 时 候 应 聘 者 应 该 要 反应 过 来 ， 面 试 官 期 竺 我 们 提供 的 不 仅仅 是 
解决 一 个 问题 的 办 法 ， 而 是 解决 一 系列 同类 型 问题 的 通用 办 法 。 这 就 是 
面试 官 在 考查 我 们 对 扩展 性 的 理解 ， 即 希望 我 们 能 够 给 出 一 个 模式 ， 在 
这 个 模式 下 能 够 很 方便 地 把 已 有 的 解决 方案 扩展 到 同类 型 的 问题 上 去 。 

回 到 面试 官 新 提出 的 两 个 问题 上 来 。 我 们 发 现 要 解决 这 两 个 新 的 问 
题 ， 其 实 只 需要 修改 函数 ReorderOddEven 中 的 两 处 判断 的 标准 ， 而 大 的 
逻辑 框架 完全 不 需要 改动 。 因 此 我 们 可 以 把 这 个 逻辑 框架 抽象 出 来 ， 而 
把 判断 的 标准 变 成 一 个 函数 指针 ， 也 束 是 用 一 个 单独 的 函数 来 判断 数字 
是 不 是 符合 标准 。 这 样 我 们 就 把 整个 函数 解 耦 成 两 部 分 : 一 是 判断 数字 
应 该 在 数组 前 半 部 分 还 是 后 半 部 分 的 标准 ， 二 是 拆 分 数组 的 操作 。 于 是 
我 们 可 以 写 出 下 面 的 代码 : 






























































void Reorder (int *pData, unsigned int length, bool (*func) (int)) 
| 
if(pData == NULL || length == 0) 
return; 


int *pBegin = pData; 
int *pEnd = pData + length - 1; 


while (pBegin < pEnd) 
{ 
while(pBegin < pEnd && !func(*pBegin) ) 
pBegin ++; 


while(pBegin < pEnd && func (*pEnd) ) 
pEnd --; 


if(pBegin < pEnd) 

{ 
int temp = *pBegin; 
*pBegin = *pEnd; 
*pEnd = temp; 


} 


bool isEven(int n) 


{ 


return (n & 1) == Q; 


在 上 面 的 代码 中 ， 函 数 Reorder 根 据 func 的 标准 把 数组 pData 分 成 两 
部 分 ， 而 函数 isEven 则 是 一 个 具体 的 标准 ， 即 判断 一 个 数 是 不 是 偶数 。 
有 了 这 两 个 函数 ， 我 们 可 以 很 方便 地 把 数组 中 的 所 有 奇数 移 到 偶数 的 前 
面 。 实 现代 码 如 下 : 
void ReorderOddEven(int *pData, unsigned int length) 


{ 


} 


Reorder (pData, length, isEven); 


如 末 把 问题 改 成 把 数组 中 的 负数 移 到 非 负 数 的 前 面 ， 或 者 把 能 被 3 
整除 的 数 移 到 不 能 和 被 3 整数 的 数 的 前 面 ， 都 只 需 定义 新 的 函数 来 确定 分 
组 的 标准 ， 而 函数 Reorder 不 需要 做 任何 改动 。 也 整 是 说 解 厢 的 好 处 就 


} 


是 提高 了 代码 的 重用 性 ， 为 功能 扩展 提供 了 便利 。 


源 代码 : 
本 题 完 整 的 源 代码 详 见 14_ReorderArray 项 目 。 
测试 用 例 : 


e 功能 测试 (输入 数组 中 的 奇数 、 侦 数 交 丛 出 现 ， 输 入 的 数组 中 
ES E E 
前 面 ) 。 

e 特殊 输入 测试 (输入 NULL 指 针 、 输 入 的 数组 只 包含 一 个 数 
本 题 考点 : 


o 考 但 应 聘 者 的 快速 思维 能 力 。 要 在 短 时 间 内 按照 要 求 把 数组 分 
隔 成 两 部 分 ， 不 是 一 件 容易 的 事情 ， 需 要 较 快 的 思维 能 力 。 

e 对 于 已 经 工作 过 儿 年 的 应 聘 者 ， 面 试 官 还 将 考查 其 对 扩展 性 的 
理解 ， 要 求 应 聘 者 写 出 的 代码 具有 可 重用 性 。 


3.4 代码 的 鲁 棒 性 


鲁 棒 是 英文 Robust 的 音译 ， 有 时 也 翻译 成 健壮 性 。 所 谓 的 鲁 棒 性 是 
和 etait 
ARTE 

容错 性 是 鲁 棒 性 的 一 个 重要 体现 。 不 鲁 棒 的 软件 在 发 生 异 常事 件 的 
时 候 ， 比 如 用 户 输入 错误 的 用 户 名 、 试 图 打开 的 文件 不 存在 或 者 网 络 不 
能 连接 ， 束 会 出 现 不 可 预见 的 诡异 行为 ， 或 者 干脆 整个 软件 骨 沉 。 这 样 
的 软件 对 于 用 户 而 言 ， 不 亚 于 一 场 灾难 。 

由 于 和 鲁 棒 性 对 软件 开发 非常 重要 ， 面 试 官 在 招聘 的 时 候 对 应 聘 者 写 
出 的 代码 是 否 鲁 棒 也 非常 关注 。 提 高 代码 的 鲁 棒 性 的 有 效 途 径 是 进行 防 
御 性 编程 。 防 御 性 编程 是 一 种 编程 习惯 ,是 指 预见 在 什么 地 方 可 能 会 出 
现 问 题 ， 并 为 这 些 可 能 出 现 的 问题 制定 处 理 方 式 。 比 如 试图 打开 文件 时 
发 现 文件 不 存在 ， 我 们 可 以 提示 用 户 检查 文件 名 和 路 径 ; 当 服 务 需 连接 
不 上 时 ， 我 们 可 以 试图 连接 备用 服务 器 等 。 这 样 当 肛 御 情况 发 生 时 ， 软 
件 的 行为 也 尽 在 我 们 的 掌握 之 中 ， 而 不 至 于 出 现 不 可 预见 的 事情 。 

在 面试 时 ， 最 简单 也 最 实用 的 防御 性 编程 就 是 在 函数 入 口 添加 代码 
以 验证 用 户 输 入 是 否 符 合 要 求 。 通 第 面试 要 求 的 是 写 一 两 个 函数 ， 我 们 














需要 格外 关注 这 些 函 数 的 输入 参数 。 如 果 输 入 的 是 一 个 指针 ， 那 指针 是 
空 指针 怎么 办 ? 如 果 输 入 的 是 一 个 字符 哩 ， 那 么 字符 串 的 内 容 为 空 怎 么 
Tp? 如 宋 能 把 这 些 问题 都 提前 考虑 到 ， 并 做 相应 的 处 理 ， 那 么 面试 官 就 
会 觉得 我 们 有 防御 性 编程 的 习惯 ， 能 够 写 出 鲁 棒 的 软件 。 

当然 并 不 是 所 有 与 鲁 棒 性 相关 的 问题 都 只 是 检查 输入 的 参数 这 么 简 
单 。 我 们 看 到 问题 的 时 候 ， 要 多 问 几 个 “如 果 不 .…... 那 么 .…...” 这 样 的 问 
题 。 比 如 面试 题 15“ 链 表 中 倒数 第 k 个 结 点 >”， 这 里 隐 含 着 一 个 条 件 就 是 
链表 中 结 反 的 个 数 大 于 k。 我 们 就 要 问 如 果 链 表 中 的 结 点 的 数目 不 是 大 
于 k 个 ， 那 么 代码 会 出 什么 问题 ?这样 的 思考 方式 能 够 帮助 我 们 发 现 潜 
在 的 问题 并 提前 解决 问题 。 这 比 让 面试 官 发 现 问题 之 后 我 们 再 去 眉 忙 分 
析 代 码 碍 找 问题 的 根源 要 好 得 多 。 


面试 题 15: 链表 中 倒数 第 kK 个 结 反 
题目 ， 输入 一 个 链表 ， 输 出 该 链表 中 倒数 第 k 个 结 点 。 为 





了 符合 大 多 数 人 的 习惯 ， 本 题 从 1 开始 计数 ， 即 链表 的 尾 结 点 
是 倒数 第 1 个 结 点 。 例 如 一 个 链表 有 6 个 结 点 ， 从 头 结 点 开始 它 
们 的 值 依次 是 1、2、3、4、5、6。 这 个 链表 的 倒数 第 3 个 结 点 
是 值 为 4 的 结 点 。 

链表 结 点 定义 如 下 : 
struct ListNode 
{ 

int m_nValue; 

ListNode* m_pNext; 


} ; 

为 了 得 到 倒数 第 k 个 结 点 ， 很 目 然 的 想法 是 先 走 到 链表 的 尾 喘 ， 再 
从 尾 病 回溯 k 步 。 可 是 我 们 从 链表 结 点 的 定义 可 以 看 出 本 题 中 的 链表 是 
单 辐 链表 ， 单 向 链表 的 结 皮 只 有 从 前 往 后 的 指针 而 没有 从 后 往 前 的 指 
针 ， 因 此 这 种 思路 行 不 通 。 

既然 不 能 从 尾 结 点 开始 遇 历 这 个 链表 ， 我 们 还 是 把 思路 回 到 头 结 点 
上 来 。 假 设 整个 链表 有 n 个 结 点 ， 那 么 倒数 第 k 个 结 点 就 是 从 头 结 点 开始 
的 第 n 一 k 十 1 个 结 点 。 如 果 我 们 能 够 得 到 链表 中 结 点 的 个 数 n， 那 我 们 只 
要 从 头 结 扣 开始 往 后 走 n 一 k 十 1 步 束 可 以 了 。 如 何 得 到 结 点 数 n? 这 个 不 
难 ， 只 需要 从 头 开 始 授 历 链表 ， 每 经 过 一 个 结 点 ， 计 数 右 加 1 束 行 了 。 

也 惑 是 说 我 们 需要 过 有 历 链 表 两 次 ， 第 一 次 统计 出 链表 中 结 点 的 个 
数 ， 第 二 次 就 能 找到 倒数 第 k 个 结 点 。 但 是 当 我 们 把 这 个 思路 解释 给 面 








试 官 之 后 ， 他 会 告诉 我 们 他 期 符 的 解法 只 需要 明 爵 链表 一 次 。 

为 了 实现 只 裔 历 链 表 一 次 残 能 找到 倒数 第 k 个 结 点 ， 我 们 可 以 定义 
两 个 指针 。 第 一 个 指针 从 链表 的 头 指针 开始 授 历 回 前 走 k 一 1， 第 二 个 指 
针 保持 不 动 ; 从 第 k 步 开始 ， 第 二 个 指针 也 开始 从 链表 的 头 指针 开 始 遍 
历 。 由 于 两 个 指针 的 距离 保持 在 k 一 1， 当 第 一 个 ( 走 在 前 面 的 ) 指针 到 
ee a ee a ee 

下 面 以 在 有 6 个 结 点 的 链表 中 找 倒 数 第 3 个 结 点 为 例 分 析 这 个 思路 的 
过 程 。 首 先 用 第 一 个 指针 从 头 结 点 开始 向 前 走 两 (2=3 一 1) 步 到 达 第 3 
个 结 点 〈 如 图 3.5《〈a) 所 示 ) 。 接 着 我 们 把 第 二 个 指针 初始 化 指 同 链 表 
的 第 一 个 结 点 《如 图 3.5 b) 所 示 ) 。 最 后 让 两 个 指针 同时 癌 前 所 历 ， 
当 第 一 个 指针 到 达 链 表 的 尾 结 点 时 ， 第 二 个 指针 指向 的 刚好 残 是 倒数 第 
3 个 结 点 《如 图 3.5 (co) 所 示 ) 。 


(a) re 
P1 

tb) EJ 
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图 3.5 ”在 有 6 个 结 点 的 链表 上 找 倒数 第 3 个 结 点 的 过 程 


TE: (a) 第 一 个 指针 在 链表 上 走 两 步 。 (b) 把 第 二 个 指针 指向 链表 的 头 结 点 。 (c) 两 
个 指针 一 同 沿 着 链表 向 前 走 。 当 第 一 个 指针 指向 链表 的 尾 结 点 时 ， 第 二 个 指针 指向 倒数 第 3 个 结 
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想 清楚 这 个 思路 之 后 ， 很 多 人 很 快 就 能 写 出 如 下 的 代码 : 


ListNode* FindKthToTail (ListNode* pListHead, unsigned int k) 
{ 

ListNode *pAhead = pListHead; 

ListNode *pBehind = NULL; 


for(unsigned int i = 0; i< k — 1; ++ i) 
I 


pAhead = pAhead->m_pNext; 


pBehind = pListHead; 
while (pAhead->m_pNext != NULL) 


pAhead = pAhead->m_pNext; 
pBehind = pBehind->m_pNext; 


return pBehind; 


有 不 少 人 在 面试 之 前 从 网 上 看 到 过 用 两 个 指针 遍历 的 思路 来 解 这 道 
el, AC BU ih eB se, fet PRES, WERE HAR 
码 。 可 是 几 天 之 后 他 们 等 来 的 不 是 Offer， 却 是 拒 信 ， 于 是 百 思 不 得 其 
解 。 其 实 原因 很 简单 ， 就 是 自己 写 的 代码 不 够 鲁 棒 。 以 上 面 的 代码 为 
例 ， 面 试 官 可 以 找 出 3 种 办 法 让 这 段 代 码 朋 误 : 

输入 的 pListHead 为 空 指针 。 由 于 代码 会 试图 访问 空 指针 指向 的 内 
F. FEJT ARo 

输入 的 以 pListHead 为 头 结 点 的 链表 的 结 点 总 数 少 于 k。 由 于 在 for 循 
环 中 会 在 链表 上 辣 前 走 k 一 1 步 ， 仍 然 会 由 于 空 指针 造成 程序 朋 误 。 

输入 的 参数 k 为 0。 由 于 Kk 是 一 个 无 符号 整数 ， 那 么 在 for 循 环 中 k 一 1 
得 到 的 将 不 是 -1， 而 是 4294967295 〈 无 符号 的 0xFFFFFFFF) 。 因 此 for 
循环 执行 的 次 数 远 远 超出 我 们 的 预计 ， 同 样 也 会 造成 程序 骨 江 。 

这 么 简单 的 代码 却 存 在 3 个 潜在 于 误 的 风险 ， 我 们 可 以 想象 当面 试 
官 看 到 这 样 的 代码 时 会 有 什么 样 的 心情 ， 最 终 他 给 出 的 是 拒 信 而 不 是 
Offer 虽 是 意料 之 外 但 也 在 情理 之 中 。 


面试 小 提示 : 


面试 过 程 中 写 代码 要 特别 注意 鲁 棒 性 。 如 果 写 出 的 代码 存在 多 处 月 溃 的 风险 ， 那 我 们 很 
有 可 能 和 Offer 失 之 交 辟 。 


针对 前 面 指出 的 3 个 问题 ， 我 们 要 分 别处 理 。 如 果 输 入 的 链表 头 指 
































针 为 NULL， 那 么 整个 链表 为 空 ， 此 时 查找 倒数 第 k 个 结 点 自然 应 该 返回 
NULL。 如 果 输 入 的 k 是 0， 也 惑 是 试图 查找 倒数 第 0 个 结 点 ， 由 于 我 们 计 
数 是 从 1 开始 的 ， 因 此 输入 0 没有 实际 意义 ， 也 可 以 返回 NULL。 如 采 链 
表 的 结 点 数 少 于 k， 在 for 循 环 中 遍历 链表 可 能 会 出 现 指向 NULL 的 
m_pNext， 因 此 我 们 在 for 循 环 中 应 该 加 一 个 t 判 晰 。 修 改 之 后 的 代码 如 
F: 

ListNode* FindKthToTail(ListNode* pListHead, unsigned int k) 
{ 





if (pListHead == NULL || k == 0) 
return NULL; 


ListNode *pAhead = pListHead; 
ListNode *pBehind = NULL; 


for (unsigned int 1 = O07 i < kK = 1 ++ i) 
{ 
if (pAhead->m_pNext != NULL) 
pAhead = pAhead->m_pNext; 


else 
{ 


return NULL; 


} 


} 


J 


pBehind = pListHead; 
while (pAhead->m_pNext != NULL) 
{ 
pAhead = pAhead->m_pNext; 
pBehind = pBehind->m_pNext; 
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return pBehind; 


源 代码 : 
本 题 完整 的 源 代 人 码 详 见 15_KthNodeFromEnd 项 目 。 
测试 用 例 : 


e 功能 测试 (第 k 个 结 反 在 链表 的 中 间 ， 第 k 个 结 点 十 链表 的 尖 结 


点 ， 第 k 个 结 点 是 链表 的 尾 结 点 ) o 
e 特殊 输入 测试 “链表 头 结 点 为 NULL 指 针 ， 链 表 的 结 点 总 数 少 
于 k，k 等 于 0) 。 


本 题 考点 : 


o 考 奋 对 链表 的 理解 。 
e 考查 代码 的 鲁 棒 性 。 重 棱 性 是 解决 这 道 题 的 关键 所 在 。 如 果 应 
A in el peer see 








相关 题目 : 


e 求 链 表 的 中 间 结 点 。 如 果 链 表 中 结 点 总 数 为 奇数 ， 返 回 中 间 结 
点 ; 如 果 结 点 总 数 是 偶数 ， 返 回 中 间 两 个 结 点 的 任意 一 个 。 为 了 解决 这 
个 问题 ， 我 们 也 可 以 定义 两 个 指针 ， 同 时 从 链表 的 头 结 点 出 发 ， 一 个 指 
针 一 次 走 一 步 ， 另 一 个 指针 一 次 走 两 步 。 当 走 得 快 的 指针 走 到 链表 的 末 
尾 时 ， 走 得 慢 的 指针 正好 在 链表 的 中 间 。 

e 判断 一 个 单身 链表 是 否 形成 了 环形 结构 。 和 前 面 的 问题 一 样 ， 
定义 两 个 指针 ， 同 时 从 链表 的 头 结 点 出 发 ， 一 个 指针 一 次 走 一 步 ， 另 一 
个 指针 一 次 走 两 步 。 如 果 走 得 快 的 指针 追 上 了 走 得 慢 的 指针 ， 那 么 链表 
就 是 环形 链表 ; 如 果 走 得 快 的 指针 走 到 了 链表 的 末尾 (m_pNext 指 向 
NULL) 都 没有 追 上 第 一 个 指针 ， 那 么 链表 就 不 是 环形 链表 。 
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当 我 们 用 一 个 指针 遍历 链表 不 能 解决 问题 的 时 候 ， 可 以 尝试 用 两 个 指针 来 遍历 链表 。 可 
以 让 其 中 一 个 指针 遍历 的 速度 快 一 些 ( 比 如 一 次 在 链表 上 走 两 步 ) ， 或 者 让 它 先 在 链表 上 走 若 
TH 












































面试 题 16: 反 转 链表 


题目 : 定义 一 个 函数 ， 输 入 一 个 链表 的 头 结 点 ， 反 转 该 链 
表 并 输出 反 转 后 链表 的 头 结 点 。 链 表 结 点 定义 如 下 : 


struct ListNode 

1 
int m_nKey; 
ListNode* m_pNext; 


解决 与 链表 相关 的 问题 总 是 有 大 量 的 指针 操作 ， 而 指针 操作 的 代码 


总 是 容易 出 错 的 。 很 多 面试 官 喜欢 出 链表 相关 的 问题 ， 就 是 想 通 过 指针 
操作 来 考查 应 聘 者 的 编码 功底 。 为 了 避免 出 错 ， 我 们 最 好 先进 行 全 面 的 
分 析 。 在 实际 软件 开发 周期 中 ， 设 计 的 时 间 通 常 不 会 比 编码 的 时 间 短 。 
在 面试 的 时 候 我 们 不 要 急于 动手 写 代 码 ， 而 是 一 开始 仔细 分 析 和 设计 ， 
这 将 会 给 面试 官 留 下 很 好 的 印象 。 与 其 很 快 写 出 一 段 漏洞 百出 的 代码 ， 
倒 不 如 仔细 分 析 再 写 出 鲁 棒 的 代码 。 

为 了 正确 地 反 转 一 个 链表 ， 需 要 调整 链表 中 指针 的 方向 。 为 了 将 调 
整 指针 这 个 复杂 的 过 程 分析 清 楚 ， 我 们 可 以 借助 图 形 来 直观 地 分 析 。 在 
图 3.6 Ca) 所 示 的 链表 中 ，h、i 利 j 是 3 个 相 邻 的 结 点 。 假 设 经 过 若干 操 
作 ， 我 们 已 经 把 结 点 h 之 前 的 指针 调整 完毕 ， 这 些 结 点 的 m_pNext 都 指 
向 前 面 一 个 结 点 。 接 下 来 我 们 把 i 的 m_pNext 指 向 h， 此 时 的 链表 结构 如 
图 3.6 (b) 所 示 。 


i) Ck] CH] 


图 3.6。 反 转 链表 中 结 点 的 m_pNext 指 针 导 致 链表 出 现 断裂 

VE: (a) 一 个 链表 。 O) 把 i 之 前 所 有 的 结 点 的 m_pNext 都 指向 前 一 个 结 点 ， 导 致 链表 
在 结 点 i、j 之 间断 裂 。 

不 难 注意 到 ， 由 于 结 点 i 的 m_pNext 指 向 了 它 的 前 一 个 结 点 ， 导 致 我 
们 无 法 在 链表 中 裔 历 到 结 点 j。 为 了 避免 链表 在 结 点 i 处 断 开 ， 我 们 需要 
在 调整 结 点 i 的 m_pNext 之 前 ， 把 结 点 j 保 存 下 来 。 

也 就 是 说 我 们 在 调整 结 点 i 的 m_pNext 指 针 时 ， 除 了 需要 知道 结 点 i 
本 和 号 之 外 ， 还 需要 i 的 前 一 个 结 点 h， 因 为 我 们 需要 把 结 点 i 的 m_pNext 指 
癌 结 点 h。 同 时 ， 我 们 还 事先 需要 保存 i 的 一 个 结 点 j， 以 防止 链表 上 断 开 。 
因此 相应 地 我 们 需要 定义 3 个 指针 ， 分 别 指向 当前 过 历 到 的 结 点 、 它 的 
前 一 个 结 点 及 后 一 个 结 点 。 

最 后 我 们 试 着 找到 反 转 后 链表 的 头 结 点 。 不 难 分 析出 反 转 后 链表 的 
头 结 点 是 原始 链表 的 尾 结 点 。 什 么 结 点 是 尾 结 点 ? 自然 是 m_pNext 为 
NULL 的 结 点 。 

有 了 前 面 的 分 析 ， 我 们 不 难 写 出 如 下 代码 : 





























ListNode* ReverseList (ListNode* pHead) 
{ 
ListNode* pReversedHead = NULL; 
ListNode* pNode = pHead; 
ListNode* pPrev = NULL; 
while(pNode != NULL) 
{ 


ListNode* pNext = pNode->m_pNext; 


if (pNext == NULL) 
pReversedHead = pNode; 


pNode->m_pNext = pPrev; 


pPrev = pNode; 
pNode = pNext; 
} 


return pReversedHead; 
} 


| 在 面试 的 过 程 中 ， 我 们 发 现 应 聘 者 的 代码 中 经 党 出 现 如 下 3 种 问 
jo: 

o 输入 的 链表 头 指针 为 NULL 或 者 整个 链表 只 有 一 个 结 点 时 ， 程 
Fe ABN ABT 

e Fa HER EIR. 

@ 返回 的 反 转 之 后 的 头 结 点 不 是 原始 链表 的 尾 结 点 。 

在 实际 面试 的 时 候 ， 不 同 应 聘 者 的 思路 各 不 相同 ， 因此 写 出 的 代码 
也 不 一 样 。 那 么 应 聘 者 如 何 才 能 及 时 发 现 并 纠正 代码 中 的 问题 ， 以 确保 
不 犯 上 述 错误 呢 ? 一 个 很 好 的 办 法 就 是 提前 想 好 测试 用 例 。 在 写 出 代码 
之 后 ， 立 即 用 事先 准备 好 的 测试 用 例 检查 测试 。 如 果 面 试 是 以 手写 代码 
的 方式 ， 那 也 要 在 心里 默默 运行 代码 做 单元 测试 。 只 有 确保 代码 通过 测 
试 之 后 ， 再 提交 面试 官 。 我 们 要 记 住 一 点 : E R 
正 问题 ， 比 在 面试 官 找 出 问题 之 后 再 去 慌 慌张 张 修改 代码 要 好 得 多 。 
SIRT Hef HN CRORES SE RCE EA ERECT LGN 
试 。 如 果 应 聘 者 能 够 想到 这 些 测试 用 例 ， 并 用 它们 来 检查 测试 目 己 的 代 
码 ， 那 就 能 保证 有 备 无 患 、 万 无 一 失 了 。 
以 这 道 题 为 例 ， 我 们 至 少 应 该 想到 几 类 测试 用 例 对 代码 做 功能 测 
WA: 


e 输入 的 链表 头 指 针 是 NULL。 














e 输入 的 链表 只 有 一 个 结 皮 。 

e 输入 的 链表 有 多 个 结 反 。 

如 果 我 们 确信 代码 能 够 通过 这 3 类 测试 用 例 的 测试 ， 那 我 们 就 有 很 
大 的 把 握 能 够 通过 这 轮 面试 了 。 





源 代码 : 
本 题 完整 的 源 代码 详 见 16_ReverseList 项 目 。 
测试 用 例 : 


o 功能 测试 (输入 的 链表 含有 多 个 结 点 ， 链 表 中 只 有 一 个 结 
点 ) 


。 特殊 输入 测试 〈 链 表 头 结 点 为 NULL 指 针 ) 。 

本 题 考点 ; 

。 考查 应 聘 者 对 链表 、 指 针 的 编程 能 力 。 

。 特别 注重 考查 应 聘 者 思维 的 全 面 性 及 写 出 来 的 代码 的 鲁 棒 性 。 
ALT IE: 

用 递归 实现 同样 的 反 转 链表 的 功能 。 


面试 题 17: 合并 两 个 排序 的 链表 


题目 ;输入 两 个 递增 排序 的 链表 ， 合 并 这 两 个 链表 并 使 新 
链表 中 的 结 点 仍然 是 按照 递增 排序 的 。 例 如 输入 图 3.7 中 的 链 
表 1 和 链表 2， 则 合并 之 后 的 升序 链表 如 链表 3 所 示 。 链 表 结 点 
JE XIIP: 
struct ListNode 


{ 





int m_ nValue; 
ListNode* m_pNext; 





图 3.7 合并 两 个 排序 链表 的 过 程 



































注 : 链表 1 和 链表 2 是 两 个 递增 排序 的 链表 ， 合 并 这 两 个 链表 得 到 升序 链表 为 链表 3。 


这 是 一 个 经 第 被 各 公司 采用 的 面试 题 。 在 面试 过 程 中 ， 我 们 友 现 应 
聘 者 最 容易 犯 两 种 错误 : 一 是 在 写 代 码 之 前 没有 对 合并 的 过 程 想 清楚 ， 
最 终 合 并 出 来 的 链表 要 么 中 间断 开 了 要 么 并 没有 做 到 递增 排序 ， 二 古代 
码 在 鲁 棒 性 方面 存在 问题 ， 程 序 一 旦 有 特殊 的 输入 《如 空 链 表 ) LSS 
沉 。 接 下 来 分 析 如 何 解 决 这 两 个 问题 。 

首先 分 析 合 并 两 个 链表 的 过 程 。 我 们 的 分 析 从 合并 两 个 链表 的 头 结 
点 开始 。 链 表 1 的 头 结 点 的 值 小 于 链表 2 的 头 结 点 的 值 ， 因 此 链表 1 的 头 
结 点 将 是 合并 后 链表 的 头 结 点 〈 如 图 3.8 Cad 所 示 ) o 








EE eir TTE E E aS 4 
a 

oy ean OTe ý 
JET m ET ES i 
| P] | 
AEE Pua r 
U f 

b — == = = = —- —— 
ı L3 盖 | 5 l] 
| | 

PI 

AEI Æ | ED 
\ P2 J 


图 3.8 合并 两 个 递增 链表 的 过 程 














TE: (Ca) 链表 1 的 头 结 点 的 值 小 于 链表 2 的 头 结 点 的 值 ， 因 此 链表 1 的 头 结 点 是 合并 后 链 
RAE. b) 在 剩余 的 结 点 中 ， 链 表 2 的 头 结 点 的 值 小 于 链表 1 的 头 结 点 的 值 ， 因 此 链表 2 
的 头 结 点 是 剩余 结 点 的 头 结 点 ， 把 这 个 结 点 和 之 前 已 经 合并 好 的 链表 的 尾 结 点 链接 起 来 。 

我 们 继续 合并 两 个 链表 中 剩余 的 结 点 《图 3.8 中 虚线 框 中 的 链 
R) 。 在 两 个 链表 中 剩 下 的 结 点 依然 是 排序 的 ， 因 此 合并 这 两 个 链表 的 
步骤 和 前 面 的 步骤 是 一 样 的 。 我 们 还 是 比较 两 个 头 结 点 的 值 。 此 时 链表 
2 的 头 结 点 的 值 小 于 链表 1 的 头绪 点 的 值 ， 因 此 链表 2 的 头 结 点 的 值 将 是 
合并 剩余 结 点 得 到 的 链表 的 头 结 点 。 我 们 把 这 个 结 点 和 前 面 合并 链表 时 
得 到 的 链表 的 尾 结 点 〈 值 为 1 的 结 点 ) 链接 起 来 ， 如 图 3.8 Cb) Ara. 

当 我 们 得 到 两 个 链表 中 值 较 小 的 头 结 点 并 把 它 链接 到 已 经 合并 的 链 
表 之 后 ， 两 个 链表 剩余 的 结 点 依然 是 排序 的 ， 因 此 合并 的 步骤 和 之 前 的 
步骤 是 一 样 的 。 这 就 是 典型 的 递归 的 过 程 ， 我 们 可 以 定义 递归 函数 完成 
这 一 合并 过 程 。 

接 下 来 我 们 来 解决 鲁 棒 性 的 问题 。 每 当代 码 试 图 访问 空 指 针 指 向 的 
内 存 时 程序 就 会 朋 尝 ， 从 而 导致 鲁 棒 性 问题 。 在 本 题 中 一 旦 输入 空 的 链 
表 就 会 引入 空 的 指针 ， 因 此 我 们 要 对 空 链 表单 独处 理 。 当 第 一 个 链表 是 
空 链表 ， 也 就 是 它 的 尖 结 点 是 一 个 空 指针 时 ， 那 么 把 它 和 第 二 个 链表 合 
并 ， 显 然 合并 的 结果 束 古 第 二 个 链表 。 同 样 ， 当 输入 的 第 二 个 链表 的 头 
结 点 是 空 指针 的 时 候 ， 我 们 把 它 和 第 一 个 链表 合并 得 到 的 结果 就 是 第 一 
个 链表 。 如 果 两 个 链表 者 是 空 链 表 ， 合 并 的 结果 是 得 到 一 个 空 链表 。 

在 我 们 想 清 楚 合 并 的 过 程 ， 并 且 知 道 哪些 输入 可 能 会 引起 鲁 棒 性 问 
题 之 后 ， 就 可 以 动手 写 代码 了 。 下 面 是 一 段 参考 代码 : 
















































































ListNode* Merge (ListNode* pHeadl, ListNode* pHead2) 


if (pHeadl == NULL) 
return pHead2; 
ije if (pHead2 == NULL) 


eturn pHead1 ; 


pMergedHead->m_ pNext = Merge (pHeadl->m pNext, pHead2); 


{ 
pMergedHead = pHead2; 
pMergedHead->m pNext = Merge(pHeadl, pHead2->m_pNext); 
} 
return pMergedHead; 
} 
源 代码 : 
本 题 完整 的 源 代码 详 见 17_MergeSortedLists 项 目 。 
测试 用 例 : 





。 马 能 测试 《输入 的 两 个 链表 有 多 个 结 点 ， 结 点 的 值 互 不 相同 或 
者 存在 值 相等 的 多 个 绪 点 ) o 

° oe AAR 
针 、 两 个 链表 中 只 有 一 个 结 点 ) 


本 题 考点 : 


© 考 碍 应聘 者 分 析 问 题 的 能 力 。 解 决 这 个 问题 需要 大 量 的 指针 操 
作 ， 应 聘 者 如 果 没 有 透彻 地 分 析 问 题 形成 清晰 的 思路 ， 那 么 他 很 难 写 出 
正确 的 代码 。 

。 考查 应 聘 者 能 不 能 写 出 鲁 棒 的 代码 。 由 于 有 大 量 指针 操作 ， 应 
聘 者 如 果 稍 有 不 慎 就 会 在 代码 中 遗留 很 多 与 鲁 棒 性 相关 的 隐患 。 建 议 应 
聘 者 在 写 代 码 之 前 全 面 分 析 哪 些 情况 会 引入 空 指针 ， 并 考虑 清楚 怎么 处 
理 这 些 空 指针 。 














面试 题 18: 树 的 子 结构 


题目 : 输入 两 棵 二 又 树 A 和 B， 判 断 B 是 不 是 A 的 子 结构 。 
二 又 树 结 点 的 定义 如 下 : 
struct BinaryTreeNode 


{ 


int m_nValue; 
BinaryTreeNode* m_pLeft; 
BinaryTreeNode* m_pRight; 


}; 
例如 图 3.9 中 的 两 棵 二 又 树 ， 由 于 A 中 有 一 部 分 子 树 的 结构 和 B 是 一 
样 的 ， 因 此 B 是 A 的 子 结构 。 








图 3.9 ”两 棵 二 义 树 A 和 B， 和 右边 的 树 B 是 左边 的 树 A 的 子 结构 


和 链表 相 比 ， 树 中 的 指针 操作 更 多 也 更 复杂 ， 因 此 与 树 相 关 的 问题 
通常 会 比 链表 的 要 难 。 如 宁 想 加 大 面试 的 难度 ， 树 的 题目 是 很 多 面试 官 
的 选择 。 面 对 着 大 量 的 指针 操作 ， 我 们 要 更 加 小 心 ， 人 否则 一 不 留神 就 会 
在 代码 中 留 下 隐患 。 

现在 回 到 这 个 题目 本 号 。 要 得 找 树 A 中 是 否 存在 和 树 B 络 构 一 样 的 
子 树 ， 我 们 可 以 分 成 两 步 : 第 一 步 在 树 A 中 找到 和 B 的 根 结 点 的 值 一 样 
ee re ene eee 

DA E TED A PY A BR EAR SP TIX SE. EEA EA 
找到 值 为 8〈 树 B 的 根 结 点 的 值 ) 的 结 点 。 从 树 A 的 根 结 点 开始 遍历 ， 我 
们 发 现 它 的 根 结 反 的 值 束 是 8。 接 着 我 们 就 去 判断 树 A 的 根 结 点 下 面 的 
子 树 是 不 是 含有 和 树 B 一 样 的 结构 〈 如 图 3.10 所 示 ) 。 在 树 A 中 ， 根 结 点 
的 左 子 结 扩 的 值 是 8， 而 树 B 的 根 结 点 的 左 子 结 点 是 9， 对 应 的 两 个 结 反 
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图 3.10 ” 树 A 的 根 结 点 和 B 的 根 结 点 的 值 相同 ， 但 树 A 的 根 结 点 下 面 “ 实 线 部 分 ) 的 结构 和 树 孔 
的 结构 不 一 致 


因此 我 们 仍然 需要 遍历 树 A， 接 着 查找 值 为 8 的 结 点 。 我 们 在 树 的 
第 二 层 中 找到 了 一 个 值 为 8 的 结 点 ， 然 后 进行 第 三 步 判 断 ， 即 判断 这 个 
结 点 下 面 的 子 树 是 否 含有 和 树 B 一 样 结构 的 子 树 〈 如 图 3.11 所 示 ) 。 于 
是 我 们 裔 历 这 个 结 点 下 面 的 子 树 ， 先 后 得 到 两 个 子 结 点 9 和 2， 这 和 树 B 
的 结构 完全 相同 。 此 时 我 们 在 树 A 中 找到 了 一 个 和 树 B 的 结构 一 样 的 子 
树 ， 因 此 树 B 是 树 A 的 子 结构 。 
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图 3.11 在 树 A 中 找到 第 二 个 值 为 8 的 结 点 ， 该 结 点 下 面 〈 实 线 部 分 ) 的 结构 和 B 的 结构 一 至 


第 一 步 在 树 A 中 查找 与 根 结 点 的 值 一 样 的 结 点 ， 这 实际 上 就 是 树 的 
衣 历 。 对 二 叉 树 这 种 数据 结构 熟悉 的 读者 自然 知道 可 以 用 吉 归 的 方法 去 
电 历 ， 也 可 以 用 循环 的 方法 去 遍历 。 由 于 递归 的 代码 实现 比较 简洁 ， 面 
和 











bool HasSubtree (BinaryTreeNode* pRootl, BinaryTreeNode* pRoot2) 


{ 


bool result = false; 
if(pRootl != NULL && pRoot2 != NULL) 
{ 
if (pRoot1l->m_nValue == pRoot2->m_nValue) 
result = DoesTreelHaveTree2 (pRootl, pRoot2); 
if (!result) 
result = HasSubtree(pRootl—->m_pLeft, pRoot2); 
if (!result) 
result = HasSubtree(pRootl->m_pRight, pRoot2); 


} 
return result; 


在 面试 的 时 候 ， 我 们 一 定 要 注意 边界 条 件 的 检查 ， 即 检查 空 指针 。 
当 树 A 或 树 B 为 空 的 时 候 ， 定 义 相 应 的 输出 。 如 果 没 有 检查 并 做 相应 的 
处 理 ， 程 序 非常 容易 骨 尝 ， 这 是 面试 时 非常 忌讳 的 事情 。 

在 上 述 代码 中 ， 我 们 递归 调用 HasSubtree 遍 历 二 叉 树 A。 如 果 发 现 
某 一 结 点 的 值 和 树 B 的 头 结 点 的 值 相同 ， 则 调用 DoesTreelHaveTree2， 
做 第 二 步 判 断 。 

第 二 步 是 判断 树 A 中 以 R 为 根 结 点 的 子 树 是 不 是 和 树 B 具 有 相同 的 结 
构 。 同 样 ， 我 们 也 可 以 用 递归 的 思路 来 考虑 : 如 果 结 点 R 的 值 和 树 B 的 
根 结 点 不 相同 ， 则 以 R 为 根 结 点 的 子 树 和 树 B 肯 定 不 具有 相同 的 结 点 ; 
如 果 它 们 的 值 相同 ， 则 递归 地 判断 它们 各 目的 无 右 结 点 的 值 是 不 是 相 
同 。 递 归 的 终止 条 件 是 我 们 到 达 了 树 A 或 者 树 B 的 叶 结 点 。 参 考 代 码 如 
F: 


} 











bool DoesTreelHaveTree2 (BinaryTreeNode* pRootl, BinaryTreeNode* 
pRoot2) 
i 
if (pRoot2 == NULL) 
return true; 


if (pRootl == NULL) 
return false; 


if (pRootl->m_nValue != pRoot2->m_nValue) 
return false; 


return DoesTreelHaveTree2 (pRooti->m_pLeft, pRoot2->m_pLeft) && 
DoesTreelHaveTree2 (pRoot1l->m_pRight, pRoot2->m_pRight) ; 


我 们 注意 到 上 述 代码 有 多 处 判断 一 个 指针 是 不 是 NULL， 这 样 做 是 
为 了 避免 试图 访问 空 指针 而 造成 程序 朋 吝 ， 同 时 也 设置 了 递归 调用 的 退 
出 条 件 。 在 写 壳 历 树 的 代码 的 时 候 一 定 要 高 度 警 惕 ， 在 每 一 处 需要 访问 
地 址 的 时 候 都 要 问 上 自己 这 个 地 址 有 没有 可 能 是 NULL， 如 果 是 NULEL 该 
怎么 处 理 。 


面试 小 提示 : 


二 叉 树 相关 的 代码 有 大 量 的 指针 操作 ， 每 一 次 使 用 指针 的 时 候 ， 我 们 都 要 问 自己 这 个 指 
针 有 没有 可 能 是 NULL， oe 

为 了 确保 自己 的 代码 完整 正确 ， 在 写 出 代码 之 后 应 聘 者 至 少 要 用 几 
个 测试 用 例 来 检验 自己 的 程序 : 树 A 和 树 B 的 头 结 点 有 一 个 或 者 两 个 都 
是 空 指针 ， 在 树 A 和 树 B 中 所 有 结 点 都 只 有 左 子 结 点 或 者 右 子 结 点 ， 树 A 
和 树 B 的 结 点 中 含有 分 又 。 只 有 这 样 才能 写 出 让 面试 官 满意 的 鲁 棒 代 
Ws 





























源 代码 : 
本 题 完 整 的 源 代 码 详 见 18_SubstructureInTree 项 目 。 
测试 用 例 : 


° a 能 测试 〈《 树 A 和 树 B 都 是 普通 的 二 又 树 ， 树 B 是 或 者 不 是 树 A 
子 结构 ) 
e 特殊 输入 测试 (两 哥 二 文 树 的 一 个 或 者 两 个 根 结 点 为 NULL 指 


针 、 二 又 树 的 所 有 结 点 都 没有 左 子 树 或 者 右 子 树 ) 。 
本 题 考点 : 


© 考 奋 对 二 又 树 通 历 算法 的 理解 及 递归 编程 能 力 。 

。 考 香 代码 的 鲁 棒 性 。 本 题 的 代码 中 含有 大 量 的 指针 操作 ， 稍 有 
不 蛋 程 序 就 会 月 证。 应 聘 者 需要 采用 防御 性 编程 的 方式 ， 每 次 访问 指针 
地 址 之 前 都 要 考虑 这 个 指针 有 没有 可 能 是 NULL。 


3.5 本章 小 结 


本 章 从 规范 性 、 完 整 性 和 和 鲁 棒 性 3 个 方面 介绍 了 如 何在 面试 时 写 出 
高 质量 的 代码 ， 如 图 3.12 所 示 。 
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图 3.12 从 规范 性 、 完 整 性 和 重 棒 性 3 个 方面 提高 代码 的 质量 


大 多 数 面 试 都 是 要 求 应 聘 者 在 白 纸 或 者 白板 上 写 代 码 。 应 聘 者 在 纺 
码 的 时 候 要 注意 规范 性 ， 尽 量 清晰 地 书写 每 个 字母 ， 通 过 缩 进 和 对 齐 括 
号 让 代码 布局 合理 ， 同 时 合理 命名 代码 中 的 变量 和 函数 。 

最 好 在 编码 之 前 全 面 考虑 所 有 可 能 的 输入 ， 确 保 写 出 的 代码 在 完成 
了 基本 功能 之 外 ， 还 考虑 了 边界 条 件 ， 并 做 好 了 错误 处 理 。 只 有 全 面 考 
虚 到 这 3 方面 的 代码 才 是 完整 的 代码 。 

为 外 ， 要 确保 自己 写 出 的 程序 不 会 轻易 骨 沉 。 平 时 在 写 代码 的 时 
候 ， 应 聘 者 最 好 养 成 防御 式 编程 的 习惯 ， 在 函数 入 口 判断 输入 是 否 有 效 























并 对 各 种 无 效 输入 做 好 相应 的 处 理 。 


PAT ”解雇 面试 题 的 思路 
4.1 面试 官 谈 面 试 思路 


人 码 前 讲 自 己 的 思路 是 一 个 考查 指标 。 一 个 合格 的 应 聘 者 应 该 在 他 做 事 之 前 明白 自己 要 
MEEA, 以 及 该 和 和 人 一 开始 就 编码 的 人 员 ， 除 非 后 面 表现 非常 优秀 ， 人 否则 很 
容易 j 
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一 一 笑 焰 《支付 宝 ， 高 级 安全 测试 工程 师 ) 
“让 应 聘 者 给 我 讲 具体 的 问题 分 析 过 程 ， 经 常会 要 求 他 证 明 。” 
— ik RA AE, ARAH) 


“个 人 比较 倾向 于 让 应 聘 者 在 写 代 码 之 前 解释 他 的 思路 。 应 聘 者 如 果 没 有 想 清 楚 束 动手 本 
| 应 聘 者 可 以 采用 举例 子 、 画 图 等 多 种 方式 ， 解 释 清楚 问题 本 身 和 问题 解雇 方案 
是 关键 。 


























































































































何 幸 杰 (SAP， 高 级 工程 师 ) 
“对 于 比较 复杂 的 算法 和 设计 ， 一 般 来 讲 最 好 是 在 开始 写 代码 前 讲 清楚 思路 和 设计 。 
一 一 狗 敏 (淘宝 ， 资 深 经 理 ) 


“喜欢 应 聘 者 先 讲 清 思 路 。 如 果 觉 察 到 方案 的 错误 和 漏洞 ， 我 会 让 他 证 明 是 否 正确 ， 主 要 
是 希望 他 能 在 分 析 的 过 程 中 发 现 这 些 错误 和 漏洞 并 加 以 改正 。” 

































































东 歼 明 (微软 ，SDE ID) 
“喜欢 应 聘 者 在 写 代 码 之 前 先 讲 思路 ， 举 例子 和 画图 都 是 很 好 的 方法 。” 
— HHE (4X, SDE II) 

















4.2 画图 让 抽象 问题 形象 化 


画图 是 在 面试 过 程 中 应 聘 者 用 来 帮助 自己 分 析 、 推理 的 常用 手段 。 
很 多 面试 题 很 抽象 ， 不 是 很 容易 找到 解决 办 法 。 这 时 不 妨 画 出 一 些 与 题 
目 相 关 的 图 形 ， 借 以 辅助 自己 观察 和 思考 。 图 形 能 使 抽象 的 问题 具体 
A 能 找到 规律 ， 从 而 找到 问题 
J 解决 方案 。 

有 不 少 与 数据 结构 相关 的 问题 ， 比 如 二 叉 树 、 二 维 数组 、 链 表 等 问 
题 ， 都 可 以 采用 画图 的 方法 来 分 析 。 很 多 时 候 衬 想 未 必 能 想 明 日 题目 中 
隐 含 的 规律 和 特点 ， 随 手 画 几 张 图 却 能 让 我 们 轻易 找到 穿 门 。 比 如 在 面 
试题 19“ 二 义 树 的 镜像 ”中 我 们 画 几 张 二 叉 树 的 图 就 能 发 现 ， 求 树 的 镜像 











的 过 程 其 实 就 是 在 过 历 树 的 同时 交换 非 叶 络 点 的 左右 子 结 点 。 在 面试 题 
20“ 顺 时 针 打 印 和 矩阵? 中， 我们 画图 之 后 很 容易 束 发 现 可 以 把 矩阵 分 解 成 
右 王 个 圆 癌 ， 然 后 从 外 回 内 打印 每 个 圆 奖 。 面 试 的 时 候 很 多 人 都 会 在 边 
界 条 件 上 犯错 误 《〈 因 为 最 后 一 圈 可 能 退化 而 不 是 一 个 完整 的 圈 ) ， 如 果 
男儿 张 示意 图 ， 束 能 够 很 容易 找到 最 后 一 疾 退 化 的 规律 。 对 于 面试 题 
26“ 复 杂 链 表 的 复制 >， 如 果 能 够 男 出 每 一 步 操作 时 的 指针 操作 ， 那 接 下 
来 写 代码 就 会 容易 得 多 。 

在 面试 的 时 候 应 聘 者 需要 加 面试 官 解释 自己 的 思路 。 对 于 复杂 的 问 
题 ， 应 聘 者 光 用 语言 未 必 能 够 说 得 清楚 。 这 个 时 候 可 以 画 出 几 个 图 形 ， 
- 边 看 看 图 形 一 边 讲解 ， 面 试 官 束 能 更 加 轻松 地 理解 应 聘 者 的 中 路 。 这 
对 应 聘 者 是 有 益 的 ， 因 为 面试 官 会 觉得 他 有 很 好 的 沟通 交流 能 力 。 


面试 题 19: 二 又 树 的 镜像 


题目 : 请 完成 一 个 函数 ， 输 入 一 个 二 又 树 ， 该 函数 输出 它 
的 镜像 。 二 又 树 结 点 的 定义 如 下 : 
struct BinaryTreeNode 


{ 





























int m_nValue; 
BinaryTreeNode* m_pLeft; 
BinaryTreeNode* m_pRight; 


} ; 
树 的 镜像 对 很 多 人 来 说 是 一 个 新 的 概念 ， 我 们 未 必 能 够 一 下 子 想 出 
求 树 的 镜像 的 方法 。 为 了 能 够 形成 直观 的 印象 ， 我 们 可 以 自己 画 一 棵 二 








又 树 ， 然 后 根据 照 镜 子 的 经 验 画 出 它 的 镜像 。 如 图 4.1 中 右边 的 二 又 树 
就 是 左边 的 树 的 镜像 。 








图 4.1 两 棵 互 为 镜像 的 二 叉 树 
仔细 分 析 这 两 棵 树 的 特点 ， 看 看 能 不 能 总 结 出 求 镜像 的 步骤 。 这 两 


标 树 的 根 结 点 相同 ， 但 它们 的 左右 两 个 子 结 点 交换 了 位 置 。 因 此 我 们 不 
妨 先 在 树 中 交换 根 结 皮 的 两 个 子 结 点 ， 束 得 到 图 4.2 中 的 第 二 标 树 。 

交换 根 结 点 的 两 个 子 结 皮 之 后 ， 我 们 注意 到 值 为 10、6 的 结 点 的 子 
结 点 仍然 保持 不 变 ， 因 此 我 们 还 需要 交换 这 两 个 结 点 的 左右 子 结 皮 。 交 
换 之 后 的 结果 分 别 是 图 4.2 中 的 第 三 柠 树 和 第 四 柠 树 。 做 完 这 两 次 交换 
之 后 ， 我 们 已 经 过 历 完 所 有 的 非 叶 子 结 点 。 此 时 变换 之 后 的 树 刚 好 就 是 
原始 树 的 镜像 。 





图 4.2 ” 求 二 又 树 镜像 的 过 程 








TE: (a) 交换 根 结 点 的 左右 子 树 : O) 交换 值 为 10 的 结 点 的 左右 子 结 点 ; 《〈c) 交换 值 
为 6 的 结 点 的 左右 子 结 点 。 


总 结 上 面 的 过 程 ， 我 们 得 出 求 一 棵 树 的 镜像 的 过 程 : 我们 先前 序 人 过 
Pees i iN ais Ue tas AL ae Re e ies 
结 点 。 当 交换 完 所 有 非 叶 子 结 点 的 左右 子 结 点 之 后 ， 就 得 到 了 树 的 镜 














像 。 

想 清 楚 了 这 个 思路 ， 我 们 就 可 以 动手 写 代 码 了。 参考 代码 如 下 : 
void MirrorRecursively (BinaryTreeNode *pNode) 
{ 

if ((pNode == NULL) || (pNode->m_pLeft == NULL && pNode->m_pRight) ) 

return; 

BinaryTreeNode *pTemp = pNode->m_ pLeft 

pNode->m pLeft = pNode->m_pRight 

pNode->m pRight = pTemp; 


if (pNode->m_pLeft) 
MirrorRecursively (pNode->m pLeft 


LE F (PNO ->m _pRigh 1 t) 
Mi DIREC Ursively (pNode->m_pRight) ; 


源 代码 
本 题 完整 的 源 代码 详 见 19_MirrorOfBinaryTree 项 目 。 
测试 用 例 


e 功能 测试 〈 普 通 的 二 叉 树 ， 二 又 树 的 所 有 结 点 都 没有 左 子 树 或 
者 右 子 树 ， 只 有 一 个 结 点 的 二 又 树 ) 。 

e 特殊 输入 测试 “二 又 树 的 根 结 点 为 NULL 指 针 ) 。 

本 题 考 点 


© 考 奋 对 二 又 树 的 理解 。 本 题 实 质 上 是 利用 树 的 过 历 算 法 解决 问 
题 。 

© 考 奋 应 聘 者 的 思维 能 力 。 树 的 镜像 是 一 个 抽象 的 概念 ， 应 聘 者 
需要 在 短 时 间 内 想 清楚 求 镜 像 的 步骤 并 转化 为 代码 。 应 聘 者 可 以 画图 把 
抽象 的 问题 形象 化 ， 这 有 助 于 其 快速 找到 解 题 思 路 。 

本 题 扩 展 


上 面 的 代码 是 用 递归 实现 的 。 如 果 要 求 用 循环 ， 该 如 何 实现 ? 
面试 题 20: MU EP FT ENE RE 


题目 : 输入 一 个 矩阵 ， 按 照 从 外 同 里 以 顺 时 针 的 顺序 依次 
打印 出 每 一 个 数字 。 例 如 : 如 果 输 入 如 下 矩阵 : 





ms 24 @ 46 

则 依次 打印 出 数字 1、2、3、4、8、12、16、15、14、13、9、5、 
6、7、11、10。 

这 道 题 完全 没有 涉及 复杂 的 数据 结构 或 者 高 级 的 算法 ， 看 起 来 是 一 
个 很 简单 的 问题 。 但 实际 上 解决 这 个 问题 ， 会 在 代码 中 包含 多 个 循环 ， 
并 且 还 需要 判断 多 个 边界 条 件 。 如 果 在 把 问题 考虑 得 很 清楚 之 前 就 开始 
写 代 码 ， 不 可 避免 会 越 写 越 混 乱 。 因 此 解决 这 个 问题 的 关键 在 于 先 要 形 
成 清晰 的 思路 ， 并 把 复杂 的 问题 分 解 成 若干 个 简单 的 问题 。 








当 我 们 遇 到 一 个 复杂 问题 的 时 候 ， 可 以 用 图 形 来 帮助 我 们 思考 。 由 
于 是 以 从 外 圈 到 内 圈 的 顺序 依次 打印 ， 我 们 可 以 把 矩阵 想象 成 春生 个 
a 我 们 可 以 用 一 个 循环 来 打印 矩阵 ， 每 一 次 打印 矩阵 
‘J — 7) Fa 
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图 4.3 ”把 矩阵 看 成 由 若干 个 顺 时 针 方 向 的 圈 组 成 


接 下 来 分 析 循 环 结束 的 条 件 。 假 设 这 个 矩阵 的 行 数 是 rows， 列 数 是 
columns。 打 印 第 一 圈 的 左上 角 的 坐标 是 〈1,1) ， 第 二 圈 的 左上 和 角 的 坐 
标 是 〈2,2) ， 依 此 类 推 。 我 们 注意 到 ， 左 上 和 角 的 坐标 中 行 标 和 列 标 总 
是 相同 的 ， 于 是 可 以 在 矩阵 中 选取 左上 角 为 (start，start〉 的 一 圈 作 为 我 
们 分 析 的 目标 。 

对 一 个 5x5 的 矩阵 而 言 ， 最 后 一 圈 只 有 一 个 数字 ， 对 应 的 坐标 为 
(2,2) 。 我 们 发 现 5>2x2。 对 一 个 6x6 的 矩阵 而 言 ， 最 后 一 图 有 4 个 数 
字 ， 其 左上 角 的 坐标 仍然 为 〈2,2) 。 我 们 发 现 6>2x2 依 然 成 立 。 于 是 我 
们 可 以 得 出 ， 让 循环 继续 的 条 件 是 columns>startXx2 并 且 
rows>startYx2。 上 所 以 我 们 可 以 用 如 下 的 循环 来 打印 矩阵 : 
































void PrintMatrixClockwisely(int** numbers, int columns, int rows) 


{ 


if (numbers == NULL || columns <= 0 || rows <= 0) 
return; 

int: start = 0; 
le(columns > start X 2 && rows > start * 2) 


while ( 


{ 


PrintMatrixInCircle(numbers, columns, rows, start); 


++start; 


} 


} 

接着 我 们 考虑 如 何 打 印 一 圈 的 功能 ， 即 如 何 实现 
PrintMatrixInCircle。 如 图 4.3 所 示 ， 我 们 可 以 把 打印 一 圈 分 为 四 步 : 第 一 
步 从 左 到 右 打 印 一 行 ， 第 二 步 从 上 到 下 打印 一 列 ， 第 三 步 从 右 到 左 打印 
一 行 ， 第 四 步 从 下 到 上 打印 一 列 。 每 一 步 我 们 根据 起 始 坐 标 和 终止 坐标 
用 一 个 循环 就 能 打印 出 一 行 或 者 一 列 。 

不 过 值得 注意 的 是 ， 最 后 一 圈 有 可 能 退化 成 只 有 一 行 、 只 有 一 列 ， 
甚至 只 有 一 个 数字 ， 因 此 打印 这 样 的 一 圈 就 不 再 需要 四 步 。 图 4.4 是 几 
个 退化 的 例子 ， 打 印 一 圈 分 别 只 需要 三 步 、 两 步 甚至 只 有 一 步 。 













































































图 4.4 打印 矩阵 最 里 面 一 图 可 能 只 需要 三 步 、 两 步 甚 至 一 步 
一 步 总 是 需要 


因此 我 们 要 仔细 分 析 打 印 时 每 一 步 的 前 提 条 件 。 第 
的 ， 因 为 打印 一 圈 至 少 有 一 步 。 如 条 只 有 一 行 ， 那 么 就 不 用 第 二 步 了 。 
也 就 是 需要 第 二 步 的 前 提 条 件 是 终止 行 号 大 于 起 始 行 号 。 需 要 第 三 步 打 
印 的 前 提 条 件 是 圈 内 至 少 有 两 行 两 列 ， 也 就 是 说 除了 要 求 终止 行 号 大 于 
起 始 行 号 之 外 ， 还 要 求 终 止 列 号 大 于 起 始 列 号 。 同 理 ， 需 要 打印 第 四 步 
的 前 提 条 件 是 至 少 有 三 行 两 列 ， 因 此 要 求 终止 行 号 比 起 始 行 号 至 少 大 
2， 同 时 终止 列 号 大 于 起 始 列 号 。 

通过 上 述 的 分 析 ， 我 们 残 可 以 写 出 如 下 代码 : 





void PrintMatrixInCircle(int** numbers, int columns, int rows, int 
start) 


{ 


int endX = columns - 1 - start; 
int endY = rows - 1 - start; 


// 从 左 到 右 打印 一 行 
for(int i = start; i <= endX; ++i) 
{ 
int number = numbers [start] [i]; 
printNumber (number) ; 
} 


// 从 上 到 下 打印 一 列 
if(start < endyY) 
{ 
for(int i = start + 1; i <= endY; ++i) 
{ 
int number = numbers [i] [endx]; 
printNumber (number) ; 


} 


// 从 右 到 左 打印 一 行 
if(start < endX && start < endy) 
{ 
for(int i = endX = 1; i >= start; --1i) 
{ 
int number = numbers [endY] [i]; 
printNumber (number) ; 


} 


// 从 下 到 上 打印 一 行 
if(start < endX && start < endY - 1) 
{ 
for (int i =“end¥ = de 1:5— Start + 1; = 
{ 
int number = numbers [i] [start]; 
printNumber (number) ; 


源 代 码 : 
本 题 完整 的 源 代码 详 见 20_PrintMatrix 项 目 。 
测试 用 例 : 


Fe nae Oe gone a 
行 一 列 。 


本 题 考点 : 


本 题 主 要 考查 应 聘 者 的 思维 能 力 。 从 外 到 内 顺 时 针 打 印 矩阵 这 个 过 
程 非常 复杂 ， 应 聘 者 如 何 能 很 快 地 找 出 其 规律 并 写 出 完整 的 代码 ， 是 解 
决 这 道 题 的 关键 。 当 问题 比较 抽象 不 容易 理解 时 ， 可 以 试 厦 夯 儿 个 图 形 
帮助 理解 ， 这 样 往往 能 更 快 地 找到 思路 。 


4.3 举例 让 抽象 问题 具体 化 


和 上 一 市 男 图 的 方法 一 样 ， 我 们 也 可 以 借助 举例 模拟 的 方法 来 思考 
分 析 复 杂 的 问题 。 当 一 眼看 不 出 问题 中 隐藏 的 规律 的 时 候 ， 我 们 可 以 试 
着 用 一 两 个 具体 的 例子 模拟 操作 的 过 程 ， 这 样 说 不 定 就 能 通过 有 具体 的 例 
子 找到 抽象 的 规律 。 比 如 面试 题 22“ 栈 的 压 入 、 弹 出 序列 ”， 很 多 人 都 不 
能 立即 找到 栈 的 压 入 和 弹出 规律 。 这 时 我 们 可 以 仔细 分 析 一 两 个 序列 ， 
一 步 一 步 模拟 压 入 、 弹 出 的 操作 ， 并 从 中 总 结 出 隐 合 的 规律 。 面 试题 
24“ 二 又 搜索 树 的 后 序 遍 历 序 列 ? 也 类 似 ， 我 们 同样 可 以 通过 一 两 个 具体 
的 序列 找到 后 续 遍 历 的 规律 。 

具体 的 例子 也 可 以 帮助 我 们 向 面试 官 解 释 算 法 思路 。 算 法 通常 是 很 
抽象 的 ， 用 语言 不 容易 表述 得 很 清楚 ， 我 们 可 以 考虑 举 出 一 两 个 具体 的 
例子 ， 告 诉 面试 官 我 们 的 算法 是 怎么 一 步 步 处 理 这 个 例子 的 。 例 如 在 面 
试题 21“ 包 含 min 函 数 的 栈 ? 中 ， 我 们 可 以 举例 模拟 压 栈 和 弹出 几 个 数 
字 ， 分 析 每 次 操作 之 后 数据 栈 、 辅 助 栈 和 最 小 值 各 是 什么 。 这 样 解释 之 
后 ， 面 试 官 就 能 很 清晰 地 理解 我 们 的 思路 ， 同 时 他 也 会 党 得 我 们 有 很 好 
的 沟通 能 力 ， 能 把 复杂 的 问题 用 很 简单 的 方式 说 清楚 。 

具体 的 例子 还 能 帮助 我 们 确保 代码 的 质量 。 在 面试 中 写 完 代 码 之 
后 ， 应 该 移 检 查 一 届 ， 确 保 没 有 问题 再 区 给 面试 官 。 怎 么 检查 呢 ? 我 们 
可 以 运行 几 个 测试 用 例 。 在 分 析 问 题 的 时 候 采 用 的 例子 就 是 测试 用 例 。 
我 们 可 以 把 这 些 例子 当做 测试 用 例 ， 在 心里 模拟 运行 ， 看 每 一 步 操作 之 
后 的 结果 和 我 们 预期 的 是 不 是 一 样 。 如 果 每 一 步 的 结果 都 和 事先 预计 的 
一 致 ， 那 我 们 就 能 确保 代码 的 正确 性 了 。 











面试 题 21: 包 售 min 函数 的 栈 


题目 : 定义 栈 的 数据 结构 ， 请 在 该 类 型 中 实现 一 个 能 够 得 
到 栈 的 最 小 元 素 的 min 函 数 。 在 该 栈 中 ， 调 用 min、push 及 pop 
的 时 间 复 杂 度 都 是 O (1) 。 

看 到 这 个 问题 ， 我 们 的 第 一 反应 可 能 是 每 次 压 入 一 个 新 元 素 进 栈 
时 ， 将 栈 里 的 所 有 元 素 排序 ， 让 最 小 的 元 素 位 于 栈 顶 ， 这 样 束 能 在 
O (1) 时 间 得 到 最 小 元 素 了 。 但 这 种 思路 不 能 保证 最 后 压 入 栈 的 元 素 
能 够 最 先 出 栈 ， 因 此 这 个 数据 结构 已 经 不 是 栈 了 。 

我 们 接着 想到 在 栈 里 添加 一 个 成 员 变 量 存 放 最 小 的 元 素 。 每 次 压 入 
一 个 新 元 素 进 栈 的 时 候 ， 如 果 该 元 素 比 当前 最 小 的 元 素 还 要 小 ， 则 更 新 
最 小 元 素 。 面 试 官 听 到 这 种 思路 之 后 就 会 问 : 如 果 当 前 最 小 的 元 素 被 弹 
出 栈 了 ， 如 何 得 到 下 一 个 最 小 的 元 素 呢 ? 

分 析 到 这 里 我 们 发 现 仅仅 添加 一 个 成 员 变 量 存放 最 小 元 素 是 不 够 
的 ， 也 就 是 说 当 最 小 元 素 被 弹出 栈 的 时 候 ， 我 们 希望 能 够 得 到 次 小 元 
素 。 因 此 在 压 入 这 个 最 小 元 素 之 前 ， 我 们 要 把 次 小 元 素 保 存 起 来 。 





















































是 不 是 可 以 把 每 次 的 最 小 元 素 〈 之 前 的 最 小 元 素 和 新 压 入 栈 的 元 素 
两 者 的 较 小 值 ) 都 保存 起 来 放 到 另外 一 个 辅助 栈 里 昵 ? 我 们 不 妨 举 几 个 
例子 来 分 析 一 下 把 元 素 压 入 或 者 弹出 栈 的 过 程 〈 如 表 4.1 所 示 ) o 
#41 栈 内 压 入 3、4、2、1 之 后 接连 两 次 弹出 栈 顶 数字 再 压 入 0 时 ， 
数据 栈 、 辅 助 栈 和 最 小 值 的 状态 




















首先 往 空 的 数据 栈 里 压 入 数字 3， 显 然 现 在 3 是 最 小 值 ， 我 们 也 把 这 
个 最 小 值 压 入 辅助 栈 。 接 下 来 往 数 据 栈 里 压 入 数字 4。 由 于 4 大 于 之 前 的 





最 小 值 ， 因 此 我 们 仍然 往 辅助 栈 里 压 入 数字 3。 第 三 步 继 续 往 数据 栈 里 
压 入 数字 2。 由 于 2 小 于 之 前 的 最 小 值 3， 因 此 我 们 把 最 小 值 更 新 为 2， 并 
把 2 压 入 辅助 栈 。 同 样 当 压 入 数字 1 时 ， 也 要 更 新 最 小 值 ， 并 把 新 的 最 小 
值 1 压 入 辅助 栈 。 

从 表 4.1 中 我 们 可 以 看 出 ， 如 果 每 次 都 把 最 小 元 素 压 入 辅助 栈 ， 那 
么 就 能 保证 辅助 栈 的 栈 顶 一 直 都 是 最 小 元 素 。 当 最 小 元 素 从 数据 栈 内 被 
弹出 之 后 ， 同 时 弹出 辅助 栈 的 栈 顶 元 素 ， 此 时 辅助 栈 的 新 栈 顶 元 素 就 是 
下 一 个 最 小 值 。 比 如 第 四 步 之 后 ， 栈 内 的 最 小 元 素 是 1。 当 第 五 步 在 数 
据 栈 内 弹出 1 后 ， 我 们 把 辅助 栈 的 栈 顶 弹出 ， 辅 助 栈 的 栈 顶 元 素 2 就 是 新 
的 最 小 元 素 。 接 下 来 继续 弹出 数据 栈 和 辅助 栈 的 栈 顶 之 后 ， 数 据 栈 还 剩 
下 3、4 两 个 数字 ，3 是 最 小 值 。 此 时 位 于 辅助 栈 的 栈 顶 数字 正好 也 是 3， 
的 确 是 最 小 值 。 这 说 明 我 们 的 思路 是 正确 的 。 

当 我 们 想 清楚 上 述 过 程 之 后 ， 就 可 以 写 代 码 了 。 下 面 是 3 个 关键 函 
数 push、pop 和 min 的 参考 代码 。 在 代码 中 ，m_data 是 数据 栈 ， 而 m_min 
是 辅助 栈 。 示 例 代码 如 下 : 




















template <typename T> void StackWithMin<T>::push(const T& value) 
{ 


m_data.push (value); 


if (m_min.size() == 0 || value < m_min.top()) 
m_min.push (value); 

else 
m_min.push(m_min.top()); 


} 


template <typename T> void StackWithMin<T>: :pop() 
{ 


assert (m_data.size() > 0 && m_min.size() > 0); 


m_data.pop(); 
m_min.pop (); 


template <typename T> const T& StackWithMin<T>::min() const 
{ 


assert (m_data.size() > 0 && m_min.size() > 0); 


return m_min.top(); 


源 代码 : 
本 题 完整 的 源 代码 详 见 21_MinInStack 项 目 。 
测试 用 例 : 
新 压 入 栈 的 数字 比 之 前 的 最 小 值 大 。 
新 压 入 栈 的 数字 比 之 前 的 最 小 值 小 。 
弹出 栈 的 数字 不 是 最 小 的 元 素 。 
弹出 栈 的 数字 是 最 小 的 元 素 。 
本 题 考点 : 
e@ 考查 分 析 复 杂 问 题 的 思维 能 力 。 在 面试 的 时 候 ， 很 多 应 聘 者 都 
止步 于 添加 一 个 变量 保存 最 小 元 素 的 思路 。 其 实 只 要 举 个 例子 多 做 几 次 


入 栈 、 出 栈 的 操作 融 能 看 出 问题 ， 并 想到 也 要 把 最 小 元 系 用 另外 的 辅助 
栈 保存 。 当 我 们 面 对 一 个 抽象 复杂 的 问题 的 时 候 ， 可 以 用 几 个 具体 的 例 

















子 来 找 出 规律 。 找 到 规律 之 后 再 解决 问题 ， 就 容易 多 了 。 
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面试 题 22: 栈 的 压 入 、 弹 出 序列 


题目 : 输入 两 个 整数 序列 ， 第 一 个 序列 表示 栈 的 压 入 顺 
序 ， 请 判断 第 二 个 序列 是 否 为 该 栈 的 弹出 顺序 。 假 设 压 入 栈 的 
所 有 数字 均 不 相等 。 例 如 序列 1、2、3、4、5 是 某 栈 的 压 栈 序 
列 ， 序 列 4、5、3、2、1 是 该 压 栈 序列 对 应 的 一 个 弹出 序列 ， 
但 4、3、5、1、2 束 不 可 能 是 该 压 栈 序列 的 弹出 序列 。 

解决 这 个 问题 很 直观 的 想法 就 是 建立 一 个 辅助 栈 ， 把 输入 的 第 一 个 
0 

弹出 数字 。 

以 弹出 序列 4、5、3、2、1 为 例 分 析 压 栈 和 弹出 的 过 程 。 第 一 个 希 
望 被 弹出 的 数字 是 4， 因 此 4 需要 先 压 入 到 辅助 栈 里 面 。 压 入 栈 的 顺序 由 
压 栈 序列 确定 了 ， 也 就 是 在 把 4 压 入 进 栈 之 前 ， 数 字 1、2、3 都 需要 先 压 
入 到 栈 里 面 。 此 时 栈 里 包含 4 个 数字 ， 分 别 是 1、2、3、4， 其 中 4 位 于 栈 
顶 。 把 4 弹出 栈 后 ， 剩 下 的 三 个 数字 是 1、2 和 3。 接 下 来 希望 被 弹出 的 数 
字 是 5， 由 于 它 不 是 栈 顶 数字 ， 因 此 我 们 接着 在 第 一 个 序列 中 把 4 以 后 数 
字 压 入 辅助 栈 中 ， 直 到 压 入 了 数字 5。 这 个 时 候 5 位 于 栈 顶 ， 就 可 以 被 弹 
出 来 了 。 接 下 来 希望 被 弹出 的 三 个 数字 依次 是 3、2 和 1。 由 于 每 次 操作 
前 它们 都 位 于 栈 顶 ， 因 此 直接 弹出 即 可 。 表 4.2 总 结 了 本 例 中 入 栈 和 出 
栈 的 步骤 。 

44.2 ” 压 栈 序列 为 IL、2、3、4、5， 弹 出 序列 4、5、3、2、1 对 应 的 压 
栈 和 弹出 过 程 
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接 下 来 青 分 析 弹 出 序列 4、3、 5、1、2。 和 第 一 个 弹出 的 数字 4 的 情况 
和 前 面 一 样 。 把 4 弹出 之 后 ，3 位 于 栈 顶 ， 可 以 直接 弹出 。 接 下 来 希望 弹 
出 的 数字 是 5， 由 于 5 不 是 栈 顶 数字 ， 到 压 栈 序列 里 把 没有 压 栈 的 数字 压 
入 辅助 栈 ， 直 至 遇 到 数字 5。 把 数字 5 压 入 栈 之 后 ，5 就 位 于 栈 顶 了 ， 可 
以 弹出 。 此 时 栈 内 有 两 个 数字 1 和 2， 其 中 2 位 于 栈 顶 。 由 于 接 下 来 需要 
弹出 的 数字 是 1， 但 1 不 在 栈 项 ， 我 们 需要 从 压 栈 序列 中 尚未 压 入 栈 的 数 
字 中 去 搜索 这 个 数字 。 但 此 时 压 栈 序列 中 所 有 数字 都 已 经 压 入 栈 了 。 所 
以 该 序列 不 是 序列 1、2、3、4、5 对 应 的 弹出 序列 。 表 4.3 总 结 了 这 个 例 
子 中 压 栈 和 弹出 的 过 程 。 
表 4.3 一 个 压 入 顺序 为 1、2、3、4、5 的 栈 没 有 一 个 弹出 序列 为 4、 

3、5、1、2 
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压 和 4 | 1,2,3,4 下 一 个 弹出 的 是 | ,但 1 不 在 栈 顶 ， 压 栈 序列 的 数字 都 已 入 栈 。 操 


[miss le wee 


总 结 上 述 入 栈 、 出 栈 的 过 程 ， 我 们 可 以 找到 判断 一 个 序列 是 不 是 栈 
的 弹出 序列 的 规律 : 如 果 下 一 个 弹出 的 数字 刚好 是 栈 顶 数字 ， 那 么 直接 
弹出 。 如 果 下 一 个 弹出 的 数字 不 在 栈 项 ， 我 们 把 压 栈 序列 中 还 没有 入 栈 





的 数字 压 入 辅助 栈 ， 直 到 把 下 一 个 需要 弹出 的 数字 压 入 栈 顶 为 止 。 如 果 
所 有 的 数字 都 压 入 栈 了 仍然 没有 找到 下 一 个 弹出 的 数字 ， 那 么 该 序列 不 
可 能 是 一 个 弹出 序列 。 

人 我 们 就 可 以 动手 写 代 码 了 。 下 面 是 一 段 参 





bool IsPopOrder(const int* pPush, const int* pPop, int nLength) 


{ 
bool bPossible = false; 


if(pPush != NULL && pPop != NULL && nLength > 0) 
{ 

const int* pNextPush = pPush; 

const int* pNextPop = pPop; 


std: :stack<int>stackData; 


while(pNextPop - pPop < nLength) 
{ 
while(stackData.empty() || stackData.top() 
{ 
if (pNextPush - pPush == nLength) 
break; 


stackData.push(*pNextPush) ; 
pNextPush ++; 

if(stackData.top() != *pNextPop) 
break; 


stackData.pop(); 
pNextPop ++; 


l= *pNextPop) 


if(stackData.empty() && pNextPop - pPop == nLength) 


bPossible = true; 


return bPossible; 


源 代码 : 


本 题 完整 的 源 代码 详 见 22_StackPushPopOrder 项 目 。 
测试 用 例 : 


e 功能 测试 (输入 的 两 个 数组 含有 多 个 数字 或 者 只 有 1 个 数字 ， 第 
二 个 数组 是 或 者 不 是 第 一 个 数组 表示 的 压 入 序列 对 应 的 栈 的 弹出 序 





列 ) 。 
e 特殊 输入 测试 (输入 两 个 NULL 指 针 )〉。 
本 题 考点 : 


e 考 香 分析 复杂 问题 的 能 力 。 刚 听 到 这 个 面试 题 的 时 候 ， 很 多 人 
可 能 都 没有 思路 。 这 个 时 候 ， 可 以 通过 举 一 两 个 例子 ， 一 步 步 分 析 压 
栈 、 弹 出 的 过 程 ， 从 中 找 出 规律 。 

o 考查 应 聘 者 对 栈 的 理解 。 


面试 题 23: 从 上 往 下 打印 二 文 树 
题目 ， 从 上 往 下 打印 出 二 叉 树 的 每 个 结 点 ， 同 一 层 的 结 点 


按照 从 左 到 右 的 顺序 打印 。 例 如 输入 图 4.5 中 的 三 又 树 ， 则 依 
次 打印 出 8、6、10、5、7、9、11。 








图 4.5 ”一 棵 二 叉 树 ， 从 上 往 下 按 层 打印 的 顺序 为 8、6、10、5、7、9、11 
二 又 树 结 点 的 定义 如 下 : 
struct BinaryTreeNode 


{ 


int m_nValue; 





BinaryTreeNode* m_pLeft; 
BinaryTreeNode* m_pRight; 


这 道 题 实质 是 考查 树 的 遍历 算法 ， 只 是 这 种 遍历 不 是 我 们 熟悉 的 前 
序 、 中 序 或 者 后 序 人 遍历。 由 于 我 们 不 太 熟 悉 这 种 按 层 表 历 的 方法 ， 可 能 
一 下 子 也 想 不 清楚 遍历 的 过 程 。 那 面试 的 时 候 怎 么 办 呢 ?” 我 们 不 妨 先 分 
析 一 下 打印 图 4.5 中 的 二 叉 树 的 过 程 。 

因为 按 层 打印 的 顺序 决定 应 该 先 打 印 根 结 点 ， 所 以 我 们 从 树 的 根 结 
点 开始 分 析 。 为 了 接 下 来 能 够 打印 值 为 8 的 结 点 的 两 个 子 结 点 ， 我 们 应 
该 在 遍历 到 该 结 点 时 把 值 为 6 和 10 的 两 个 结 点 保存 到 一 个 容器 里 ， 现 在 
容器 内 就 有 两 个 结 点 了 。 按 照 从 左 到 右 打 印 的 要 求 ， 我 们 先 取 出 值 为 6 
的 结 点 。 打 印 出 值 6 之 后 把 它 的 值 分 别 为 5 和 7 的 两 个 结 点 放 入 数据 容 
器 。 此 时 数据 容器 中 有 三 个 结 点 ， 值 分 别 为 10、5 和 7。 接 下 来 我 们 从 数 
据 容 器 中 取出 值 为 10 的 结 点 。 注 意 到 值 为 10 的 结 点 比值 为 5、7 的 结 点 先 
放 入 容器 ， 此 时 又 比 这 两 个 结 点 先 取 出 ， 这 就 是 我 们 通常 说 的 先入 先 
出 ， 因 此 不 难看 出 这 个 数据 容器 应 该 是 一 个 队列 。 由 于 值 为 5、7、9、 
11 的 结 点 都 没有 子 结 点 ， 因 此 只 要 依次 打印 即 可 。 整 个 打印 过 程 如 表 
4.4 所 示 。 


#44 按 层 打 印 图 4.5 中 的 二 又 树 的 过 程 


网 
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打印 结 点 6 结 点 10、 结 点 5、 结 点 7 





3 打印 结 点 10 结 点 5、 结 点 7、 结 点 9、 结 点 1 
4 打印 结 点 5 结 点 7、 结 点 9、 结 点 1] 

打印 结 点 7 结 点 9、 结 点 1 

打印 结 点 9 结 点 11 
7 打印 结 点 11 


通过 上 面具 体例 子 的 分 析 ， 我 们 可 以 找到 从 上 到 下 打印 三 又 树 的 规 
律 : 每 一 次 打印 一 个 结 点 的 时 候 ， 如 果 该 结 点 有 子 结 点 ， 则 把 该 结 反 的 
子 结 点 放 到 一 个 队列 的 末尾 。 接 下 来 到 队列 的 头 部 取出 最 早 进 入 队列 的 
人 eran E SA 


”既然 我 们 已 经 确定 数据 容器 是 一 个 队列 了 ， 现 在 的 问题 就 是 如 何 实 





现 队 列 。 实 际 上 我 们 无 须 自己 动手 实现 ， 因 为 STL 已 经 为 我 们 实现 了 一 


人 
RID: 


void PrintFromTopToBottom(BinaryTreeNode* pTreeRoot) 
{ 


if (!pTreeRoot) 
return; 


std: :deque<BinaryTreeNode *> dequeTreeNode; 
dequeTreeNode.push_back (pTreeRoot) ; 


while (dequeTreeNode.size()) 


{ 


BinaryTreeNode *pNode = dequeTreeNode.front (); 
dequeTreeNode.pop_front(); 


printf ("td ", pNode->m_nValue) ; 


if (pNode->m_pLeft) 


dequeTreeNode.push_back (pNode->m_pLeft) ; 


if (pNode->m_pRight) 


dequeTreeNode.push_back (pNode->m_pRight) ; 


本 题 考点 : 


e ZABER. HAM EB Pi OM, XIR E Ee 
CNBR AS» TERLAN Te] AY AB a AE A ce PE So HS 
者 通过 具体 的 例子 找 出 其 中 的 规律 并 想到 基于 队列 的 算法 ， 是 解决 这 个 
问题 的 关键 所 在 。 

© 考 奋 应 聘 者 对 二 又 树 及 队列 的 理解 。 


本 题 扩展 : 


如 何 广 虐 优先 吉 历 一 个 有 向 图 ?这 同样 也 可 以 基于 队列 实现 。 树 是 
图 的 一 种 特殊 退化 形式 ， 从 上 到 下 按 层 过 历 二 又 树 ， 从 本 质 上 来 说 就 是 








广度 优先 过 历 二 叉 树 。 
举一反三 : 


不 管 是 广度 优先 饥 历 一 个 有 向 图 还 是 一 棵 树 ， 都 要 用 到 队列 。 第 一 步 我 们 把 起 始 结 点 
《对 树 而 言 是 根 结 点 ) 放 入 队列 中 。 接 下 来 每 一 次 从 队列 的 头 部 取出 一 个 结 点 ， 过 历 这 个 结 点 
之 后 把 从 它 能 到 达 的 结 点 〈 对 树 而 言 是 子 结 点 ) 都 依次 放 入 队列 。 我 们 重复 这 个 裔 历 过 程 ， 直 
到 队列 中 的 结 点 全 部 被 过 历 为 止 。 

































































面试 题 24: 二 又 搜索 树 的 后 序 壳 有 历 序列 


题目 : 输入 一 个 整数 数组 ， 判 断 该 数组 是 不 是 某 二 又 搜索 
树 的 后 序 届 历 的 结果 。 如 果 是 则 返回 true， 人 否则 返回 false。 假 
设 输入 的 数组 的 任意 两 个 数字 都 互 不 相同 。 

例如 输入 数组 {5,7,6,9,11,10,8}， 则 返回 true， 因 为 这 个 整数 序列 是 
图 4.6 二 又 搜索 树 的 后 序 遍 历 结 果 。 如 果 输 入 的 数组 是 {7,4,6,5}， 由 于 没 
有 哪 棵 二 又 搜索 树 的 后 序 遍 历 的 结果 是 这 个 序列 ， 因 此 返回 false。 




















图 4.6 ”后 序 遍历 序 列 5、7、6、9、11、10、8 对 应 的 二 又 搜索 树 


在 后 序 遍 历 得 到 的 序列 中 ， 最 后 一 个 数字 是 树 的 根 结 点 的 值 。 数 组 
中 前 面 的 数字 可 以 分 为 两 部 分 : 第 一 部 分 是 左 子 树 结 点 的 值 ， 它 们 都 比 
根 结 点 的 值 小 ， 第 二 部 分 是 右 子 树 结 点 的 值 ， 它 们 都 比 根 结 点 的 值 大 。 

以 数组 {5,7,6,9,11,10,8} 为 例 ， 后 序 壳 历 结 果 的 最 后 一 个 数字 8 就 是 
根 结 点 的 值 。 在 这 个 数组 中 ， 前 3 个 数字 5、7 和 6 都 比 8 小 ， 是 值 为 8 的 结 
后 3 个 数字 9、11 和 10 都 比 8 大 ， 是 值 为 8 的 结 点 的 右 子 
X EA hyo 

我 们 接 下 来 用 同样 的 方法 确定 与 数组 每 一 部 分 对 应 的 子 树 的 结构 。 
这 其 实 就 是 一 个 递归 的 过 程 。 对 于 序列 5、7、6， 最 后 一 个 数字 6 是 左 子 
树 的 根 结 点 的 值 。 数 字 5 比 6 小 ， 是 值 为 6 的 结 点 的 左 子 结 点 ， 而 7 则 是 它 


























的 右 子 结 点 。 同 样 ， 在 序列 9、11、10 中 ， 最 后 一 个 数字 10 是 右 子 树 的 
aa 数字 9 比 10 小 ， 是 值 为 10 的 结 点 的 左 子 结 点 ， 而 11 则 是 它 的 右 
Ares 
我 们 再 来 分 析 另 一 个 整数 数组 {7,4,6,5}。 后 序 遍 历 的 最 后 一 个 数 是 

根 结 点 ， 因 此 根 结 点 的 值 是 s。 由 于 第 一 个 数字 7 大 于 5， 因 此 在 对 应 的 
二 叉 搜 索 树 中 ， 根 结 点 上 是 没有 左 子 树 的 ， 数 字 7、4 和 6 都 是 右 子 树 结 
点 的 值 。 但 我 们 发 现在 右 子 树 中 有 一 个 结 点 的 值 是 4， 比 根 结 点 的 值 5 
小 ， 这 违背 了 二 又 搜索 树 的 定义 。 因 此 不 存在 一 棵 二 又 搜索 树 ， 它 的 后 
序 通 历 的 结果 是 7、4、6、5。 

oe ee 就 不 是 一 件 很 困难 的 事情 了 。 下 面 是 参 





bool VerifySquenceOfBST(int sequence[], int length) 
{ 
if (sequence == NULL || length <= 0) 
return false; 


int root = sequence[length - 1]; 


// 在 二 又 搜索 树 中 左 子 树 的 结 点 小 于 根 结 点 
int i: =; 
fort; 1 3 length = lp ++ i) 
{ 
if (sequence[i] > root) 
break; 


} 
// 在 二 又 搜索 树 中 右 子 树 的 结 点 大 于 根 结 点 


int j = i; 
for{; j < length = 1; ++ j) 
{ 

if (sequence[j] < root) 

return false; 

} 
// 判断 左 子 树 是 不 是 二 又 搜索 树 
bool left = true; 
if(i > 0) 

left = VerifySquenceOfBST (sequence, i); 


// 判断 右 子 树 是 不 是 二 又 搜索 树 
bool right = true; 
if(i < length - 1) 
right = VerifySquenceOfBST (sequence + i, length - i - 1); 


return (left && right); 


源 代码 : 
本 题 完整 的 源 代码 详 见 24_SquenceOfBST 项 目 。 
测试 用 例 : 


e 功能 汕 试 〈 输 入 的 后 序 遇 历 的 序列 对 应 一 柠 二 又 树 ， 包 括 完 全 


二 叉 树 、 所 有 结 点 都 没有 左 / 右 子 树 的 二 又 树 、 只 有 一 个 结 点 的 二 又 
BY; 输入 的 后 序 过 有 历 的 序列 没有 对 应 一 柠 二 又 树 ) 。 

。 特殊 输入 测试 〈 指 向 后 序 遍历 序列 的 指针 为 NULL 指 针 ) 。 

本 题 考点 : 

。 考查 分 析 复 杂 问 题 的 思维 能 力 。 能 否 解决 这 道 题 的 关键 在 于 应 
聘 者 是 否 能 找 出 后 序 遍 历 的 规律 。 一 旦 找到 规律 了 ， 用 递归 的 代码 编码 
相对 而 言 束 简单 了 。 在 面试 的 时 候 ， 应 聘 者 可 以 从 一 两 个 例子 入 手 ， 通 
过 分 析 具 体 的 例子 寻找 规律 。 

© 考 香 对 二 又 树 后 序 遍 历 的 理解 。 

相关 题目 : 

输入 一 个 整数 数组 ， 判 断 该 数组 是 不 是 菜 二 又 搜索 树 的 前 序 过 历 的 
结 末 。 这 和 前 面 问题 的 后 序 通 历 很 类 似 ， 只 是 在 前 序 遇 历 得 到 的 序列 
中 ， 第 一 个 数字 是 根 结 点 的 值 。 

n 

如 果 面 试题 是 要 求 处 理 一 棵 二 又 树 的 遍历 序列 ， 我 们 可 以 先 找 到 二 又 树 的 根 结 点 ， 再 基 


于 根 结 点 把 整 棵 树 的 遍历 序列 拆 分 成 左 子 树 对 应 的 子 序 列 和 右 子 树 对 应 的 子 序列 ， 接 下 来 再 递 
归 地 处 理 这 两 个 子 序列 。 本 面试 题 是 应 用 这 个 思路 ， 面 试题 6 重建 二 又 树 ” 也 是 应 用 这 个 思路 。 


面试 题 25: 二 又 树 中 和 为 东 一 值 的 路 径 














































































































题目 ;输入 一 棵 二 又 树 和 一 个 整数 ， 打 印 出 二 又 树 中 结 点 
值 的 和 为 输入 整数 的 所 有 路 径 。 从 树 的 根 结 点 开始 往 下 一 直到 
叶 结 点 所 经 过 的 结 氮 形成 一 条 路 径 。 二 又 树 结 点 的 定义 如 下 : 
struct BinaryTreeNode 
{ 

int m_nValue; 

BinaryTreeNode* m_pLeft; 

BinaryTreeNode* m_pRight; 


A- 
AURA 14.78 = SOE 2, MEE RK, Ea 
径 包 含 结 点 10、12， 第 二 条 路 径 包 含 结 点 10、5 和 7。 











图 4.7 二 叉 树 中 有 两 条 和 为 22 的 路 径 : 一 条 路 径 经 过 结 点 10、5、7， 另 一 条 路 径 经 过 结 点 
10、12 


一 般 的 数据 结构 和 算法 的 教材 都 没有 介绍 树 的 路 径 ， 因 此 对 大 多 数 
应 聘 者 而 言 ， 这 是 一 个 新 概念 ， 也 就 很 难 一 下 子 想 出 完整 的 解 题 思 路 。 
这 个 时 候 我 们 可 以 试 着 从 一 两 个 具体 的 例子 入 手 ， 找 到 规律 。 

以 图 4.7 的 二 又 树 作 为 例子 来 分 析 。 由 于 路 径 是 从 根 结 点 出 发 到 叶 
结 点 ， 也 就 是 说 路 径 总 是 以 根 结 点 为 起 始点 ， 因 此 我 们 首先 需要 遍历 根 
结 点 。 在 树 的 前 序 、 中 序 、 后 序 三 种 遍历 方式 中 ， 只 有 前 序 遍 历 是 首先 
访问 根 结 点 的 。 

按照 前 序 遍 历 的 顺序 遍历 图 4.7 中 的 二 又 树 ， 在 访问 结 点 10 之 后 ， 
就 会 访问 结 点 5。 从 二 又 树 结 点 的 定义 可 以 看 出 ， 在 本 题 的 二 叉 树 结 点 
中 没有 指向 父 结 点 的 指针 ， 访 问 到 结 点 5 的 时 候 ， 我 们 是 不 知道 前 面 经 
过 了 哪些 结 点 的 ， 除 非 我 们 把 经 过 的 路 径 上 的 结 点 保存 下 来 。 每 访问 到 
一 个 结 点 的 时 候 ， 我 们 都 把 当前 的 结 点 添加 到 路 径 中 去 。 到 达 结 点 5 
时 ， 路 径 中 包含 两 个 结 点 ， 它 们 的 值 分 别 是 10 和 5。 接 下 来 过 历 到 结 点 
4， 我 们 把 这 个 结 点 也 添加 到 路 径 中 。 这 个 时 候 已 经 到 达 了 叶 结 点 ， 但 
路 径 上 三 个 结 点 的 值 之 和 是 19。 这 个 和 不 等 于 输入 的 值 22， 因 此 不 是 符 
合 要 求 的 路 径 。 

我 们 接着 要 遍历 其 他 的 结 点 。 在 遍历 下 一 个 结 点 之 前 ， 先 要 从 结 点 
4 回 到 结 点 5， 再 去 遍历 结 点 5 的 右 子 结 点 7。 值 得 注意 的 是 ， 回 到 结 点 5 
的 时 候 ， 由 于 结 点 4 已 经 不 在 前 往 结 点 7 的 路 径 上 了 ， 我 们 需要 把 结 点 4 
从 路 径 中 删除 。 接 下 来 访问 到 结 点 7 的 时 候 ， 再 把 该 结 点 添加 到 路 径 
中 。 此 时 路 径 中 三 个 结 点 10、5、7 之 和 刚好 是 22， 是 一 条 符合 要 求 的 路 
径 。 

我 们 最 后 要 遍历 的 结 点 是 12。 在 遍历 这 个 结 点 之 前 ， 需 要 先 经 过 结 
点 5 回 到 结 点 10。 同 样 ， 每 一 次 当 从 子 结 点 回 到 父 结 点 的 时 候 ， 我 们 都 
需要 在 路 径 上 删除 子 结 点 。 最 后 从 结 点 10 到 达 结 点 12 的 时 候 ， 路 径 上 的 
两 个 结 点 的 值 之 和 也 是 22， 因 此 这 也 是 一 条 符合 条 件 的 路 径 。 

我 们 可 以 用 表 4.5 总 结 上 述 的 分 析 过 程 。 

#45 ”遍历 图 4.7 中 的 二 又 树 的 过 程 
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分 析 完 HU Ea a 些 规 律 。 当 用 前 序 
历 的 方式 访问 到 某 一 结 点 时 ， 我 们 把 该 结 所 添加 到 路 2E, o 
点 的 值 。 如 果 该 结 点 为 叶 结 点 并 且 路 径 中 结 扣 值 的 和 刚好 等 于 输入 的 束 
数 ， 则 当前 的 路 径 符合 要 求 ， 我 们 把 它 打印 出 来 。 如 果 当 前 结 点 不 是 叶 
oe A, M RI ERTA. 4A A e2 BAUE, IRO AZ 
回 到 它 的 父 结 点 。 因此 我 们 在 函数 退出 之 前 要 在 路 至 上 删除 当前 结 点 并 
减 去 当前 结 oe 以 确保 返回 父 结 点 时 路 径 刚 好 是 从 根 结 点 到 父 结 点 
的 路 径 。 我 们 不 难看 出 保存 路 径 的 数据 吉 构 实际 上 是 一 个 栈 ， 因 为 路 径 
a a 而 递归 调用 的 本 质 就 是 一 个 压 栈 和 出 栈 的 过 
Eo 
, 形成 了 清晰 的 思路 之 后 ， 就 可 以 动手 写 代码 了 。 下 面 是 一 段 参考 代 




















void FindPath(BinaryTreeNode* pRoot, int expectedSum) 
{ 
if (pRoot == NULL) 
return; 


std::vector<int> path; 

int currentSum = 0; 

FindPath(pRoot, expectedSum, path, currentSum); 
} 


void FindPath 
( 
BinaryTreeNode* pRoot, 


int expectedSum, 
std: :vector<int>& path, 
inté& currentSum 


currentSum += pRoot-—>m_nValue; 
path.push_back (pRoot->m_nValue) ; 


// 如 果 是 叶 结 点 ， 并 且 路 径 上 结 点 的 和 等 于 输入 的 值 
// 打印 出 这 条 路 径 
bool isLeaf = pRoot->m_pLeft == NULL && pRoot->m_pRight == NULL; 
if (currentSum == expectedSum && isLeaf) 
{ 

printf("A path is found: "); 

std::vector<int>::iterator iter = path.begin(); 

for(; iter != path.end(); ++ iter) 

PLINEE (FANE™,. SITEET 


printt(*\n*); 


// 如 果 不 是 叶 结 点 ， 则 遍历 它 的 子 结 点 
if (pRoot->m_pLeft != NULL) 

FindPath (pRoot->m_pLeft, expectedSum, path, currentSum); 
if (pRoot->m_pRight != NULL) 

FindPath (pRoot->m_pRight, expectedSum, path, currentSum); 


// 在 返回 到 父 结 点 之 前 ， 在 路 径 上 删除 当前 结 点 ， 
// 并 在 currentSurm 中 减 去 当前 结 点 的 值 
currentSum -= pRoot->m_nValue; 
path.pop_back (); 


在 前 面 的 代码 中 ， 我 们 用 标准 模板 库 中 的 vector 实 现 了 一 个 栈 来 保 
存 路 径 ， 每 一 次 都 用 push_back 在 路 径 的 末尾 添加 结 点 ， 用 pop_back 在 路 
径 的 末尾 删除 结 点 ， 这 样 就 保证 了 栈 的 先入 后 出 的 特性 。 这 里 没有 直接 
用 STL 中 的 stack 的 原因 是 在 stack 中 只 能 得 到 栈 顶 元 素 ， 而 我 们 打印 路 径 
的 时 候 需 要 得 到 路 径 上 的 所 有 结 点 ， 因 此 在 代码 实现 的 时 候 std::stack 不 
是 最 好 的 选择 。 


源 代码 : 
本 题 完整 的 源 代 码 详 见 25_PathInTree 项 目 。 
测试 用 例 : 


e 功能 汕 试 〈 二 又 树 中 有 一 条 、 多 条 符合 条 件 的 路 径 ， 二 又 树 中 
没有 符合 条 件 的 路 径 ) 。 
e 特殊 输入 测试 〈 指 回 二 又 树 根 络 点 的 指针 为 NULIL 指 针 ) 。 


本 题 考点 : 


。 考 碍 分 析 复 杂 问 题 的 思维 能 力 。 应 聘 者 遇 到 这 个 问题 的 时 候 ， 
如 末 一 下 子 没有 思路 ， 不 妨 从 一 个 具体 的 例子 开始 ， 一 步 步 分 析 路 径 上 
包含 哪些 结 点 ， 这 样 就 能 找 出 其 中 的 规律 ， 从 而 想到 解决 方案 。 

。 考 香 对 二 又 树 的 前 序 壳 历 的 理解 。 


44 分解 让 复杂 问题 简单 化 


很 多 读者 可 能 都 知道 “各 个 击破 ”的 军事 思想 ， 这 种 思想 的 精 艇 是 当 
政 我 实力 悬殊 时 ， 我 们 可 以 把 强大 的 敌人 分 割 开 来 ， 然 后 集中 优势 兵力 
打败 被 分 割 开 来 的 小 部 分 政和 人 。 要 一 下 子 战胜 总 体 很 强大 的 政和 人 很 困 
难 ， 但 战胜 小 股 敌 人 就 容易 多 了 。 同 样 ， 在 面试 中 当 我 们 过 到 复杂 的 大 
问题 的 时 候 ， 如 果 能 够 先 把 大 问题 分 解 成 硅 干 个 简单 的 小 问题 ， 然 后 再 
逐个 解决 这 些小 问题 ， 那 可 能 也 会 容易 很 多 。 

我 们 可 以 按照 解决 问题 的 步骤 来 分 解 复 杂 问 题 ， 每 一 步 解决 一 个 小 
问题 。 比 如 在 面试 题 26“ 复 杂 链 表 的 复制 ?中 ， 我 们 将 复杂 链表 复制 的 过 
程 分 解 成 三 个 步骤 。 在 写 代 码 的 时 候 我 们 为 每 一 步 定 义 一 个 函数 ， 这 样 
每 个 函数 完成 一 个 功能 ， 整 个 过 程 的 逻辑 也 就 非常 清晰 明了 了 。 

在 计算 机 领域 有 一 类 算法 叫 分 治 法 ， 即 “分 而 治之 ”， 采 用 的 就 是 各 
个 击破 的 思想 。 我 们 把 分 解 之 后 的 小 问题 各 个 解决 ， 然 后 把 小 问题 的 解 
决 方 采 结合 起 来 解决 大 问题 。 比 如 面试 题 27“ 二 又 搜 索 树 与 双向 链 


表 ” 中 ， 转 换 整 个 二 又 树 是 一 个 大 问题 ， 我 们 先 把 这 个 大 问题 分 解 成 转 
换 左 子 树 和 右 子 树 两 个 小 问题 ， 然 后 再 把 转换 左右 子 树 得 到 的 链表 和 根 
人 
人 码 实 现 。 

在 面试 题 28“ 字 符 串 的 排列 * 中 ， 我 们 把 整个 字符 串 分 为 两 部 分 : 第 
一 个 字符 及 它 后 面 的 所 有 字符 。 我 们 先 拿 第 一 个 字符 和 后 面 的 每 个 字符 
交换 ， 交 换 之 后 再 求 后 面 所 有 字符 的 排列 。 整 个 字符 串 的 排列 是 一 个 大 
问题 ， 那 么 第 一 个 字符 之 后 的 字符 串 的 排列 就 是 一 个 小 问题 。 因 此 这 实 
际 上 也 是 分 治 法 的 应 用 ， 可 以 用 递归 实现 。 


面试 题 26: 复杂 链表 的 复制 


题目 : 请 实现 函数 ComplexListNode- 
Clone (ComplexListNode’ pHead) ， 复 制 一 个 复杂 链表 。 在 复 
杂 链 表 中 ， 每 个 结 点 除了 有 一 个 m_pNext 指 针 指 同 下 一 个 结 点 
外 ， 还 有 一 个 m_pSibling 指 癌 链 表 中 的 任意 结 点 或 者 NULL。 
结 点 的 C++ 定义 如 下 : 








struct ComplexListNode 

{ 
int m nValue; 
Comp lexListNode* m pNext; 
ComplexListNode* m pSibling; 


lat 

14.878“ Sa IY 5S AS HE eo HSE ii Ske 7Nm_pNext 
Het, 虚线 箭头 表示 m_pSibling 指 针 。 为 简单 起 见 ， 指 加 NULL 的 指针 
没有 男 出 。 
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图 4.8 一 个 含有 5 个 结 点 的 复杂 链表 


TE: 在 复杂 链表 的 结 点 中 ， 除 了 有 指向 下 一 结 点 的 指针 《〈 实 线 箭头 ) 外 ， 还 有 指向 任意 
结 点 的 指针 《虚线 箭头 ) o 


听 到 这 个 问题 之 后 ， 很 多 应 聘 者 的 第 一 反应 是 把 复制 过 程 分 成 两 









































步 : 第 一 步 是 复制 原始 链表 上 的 每 一 个 结 点 ， 并 用 m_pNext 链 接 起 来 ; 
第 二 步 是 设置 每 个 结 点 的 m_pSibling 指 针 。 假 设 原 始 链表 中 的 茶 个 结 点 
N 的 m_pSibling 指 向 结 点 S$， 由 于 S$ 的 位 置 在 链表 中 可 能 在 N 的 前 面 也 可 
能 在 N 的 后 面 ， 所 以 要 定位 S 的 位 置 需要 从 原始 链表 的 头 结 点 开始 找 。 
如 果 从 原始 链表 的 头 结 点 开始 沿 着 m_pNext 经 过 s 步 找到 结 点 S， 那 么 在 
复制 链表 上 结 点 N 的 m_pSibling 〈 记 为 S') 离 复制 链表 的 头 结 点 的 距离 
也 是 沿 着 m_pNext 指 针 s 步 。 用 这 种 办 法 我 们 就 可 以 为 复制 链表 上 的 每 个 
结 点 设置 m_pSibling 指 针 。 

对 于 一 个 含有 n 个 结 点 的 链表 ， 由 于 定位 每 个 结 点 的 m_pSibling 都 需 
要 从 链表 头 结 点 开始 经 过 O (n) 步 才 能 找到 ， 因 此 这 种 方法 的 总 时 间 
复杂 度 是 O m). 

由 于 上 述 方法 的 时 间 主 要 花费 在 定位 结 点 的 m_pSibling 上 面 ， 我 们 
试 着 在 这 方面 去 做 优化 。 我 们 还 是 分 为 两 步 : 第 一 步 仍 然 是 复制 原始 链 
表 上 的 每 个 结 点 N 创 建 N'， 然 后 把 这 些 创建 出 来 的 结 点 用 m_pNext 链 接 
起 来 。 同 时 我 们 把 <N，N'"> 的 配对 信息 放 到 一 个 哈 希 表 中 。 第 二 步 还 是 
设置 复制 链表 上 每 个 结 点 的 m_pSibling。 如 果 在 原始 链表 中 结 点 N 的 
m_pSibling 指 向 结 点 $5， 那么 在 复制 链表 中 ， 对 应 的 N’? 应 该 指向 S*。 由 于 
有 了 哈 希 表 ， 我 们 可 以 用 O (1) 的 时 间 根 据 S 找 到 S’。 

第 二 种 方法 相当 于 用 空间 换 时 间 。 对 于 有 n 个 结 点 的 链表 我 们 需要 
一 个 大 小 为 O Ch) 的 哈 希 表 ， 也 就 是 说 我 们 以 O(n) 的 空间 消耗 把 时 
间 复 杂 度 由 O Gm) 降低 到 O Cn) 。 

接 下 来 我 们 再 换 一 种 思路 ， 在 不 用 辅助 空间 的 情况 下 实现 O_ Cn) 
的 时 间 效 率 。 第 三 种 方法 的 第 一 步 仍 然 是 根据 原始 链表 的 每 个 结 点 N 创 
建 对 应 的 N'。 这 一 次 ， 我 们 把 N" 链 接 在 N 的 后 面 。 图 4.8 的 链表 经 过 这 一 
步 之 后 的 结构 ， 如 图 4.9 所 示 。 





























图 4.9 复制 复杂 链表 的 第 一 步 
VE: 复制 原始 链表 的 任意 结 点 N 并 创建 新 结 点 N\， 再 把 N 链 接 到 N 的 后 面 。 
完成 这 一 步 的 代码 如 下 : 






































void CloneNodes (ComplexListNode* pHead) 


{ 
ComplexListNode* pNode = pHead; 


while (pNode != NULL) 

{ 
ComplexListNode* pCloned = new ComplexListNode (); 
pCloned->m_nValue = pNode->m_nValue; 
pCloned->m_pNext = pNode->m_pNext; 


pCloned->m_pSibling = NULL; 
pNode->m_pNext = pCloned; 


pNode = pCloned->m_pNext; 


} 


第 二 步 设 置 复制 出 来 的 结 点 的 m_pSibling。 假 设 原始 链表 上 的 N 的 
m_pSibling 指 向 结 点 $5， 那么 其 对 应 复制 出 来 的 N’? 是 N 的 m_pNext 指 癌 的 
结 点 ， 同 样 $; 也 是 S 的 m_pNext 指 同 的 结 点 。 设 置 m_pSibling 之 后 的 链表 
如 图 4.10 所 示 。 


} 

















图 4.10 复制 复杂 链表 的 第 二 步 











VE: 如 果 原 始 链 表 上 的 结 点 N 的 m_pSibling 指 向 S， 则 它 对 应 的 复制 结 点 N' 的 m_pSibling 指 
向 S 的 下 一 结 点 S'。 


下 面 是 完成 第 二 步 的 参考 代码 ; 








void ConnectSiblingNodes (ComplexListNode* pHead) 
{ 
ComplexListNode* pNode = pHead; 
while (pNode != NULL) 
{ 
ComplexListNode* pCloned = pNode->m_pNext; 
if (pNode->m_pSibling != NULL) 
{ 
pCloned->m_pSibling = pNode->m_pSibling->m_pNext; 
} 
pNode = pCloned->m_pNext; 


} 


第 三 步 把 这 个 长 链表 拆 分 成 两 个 链表 : 把 奇数 位 置 的 结 点 用 
m_pNext 链 接 起 来 就 是 原始 链表 ， 把 偶数 位 置 的 结 点 用 m_pNext 链 接 起 
ZN o 


} 

















图 4.11 复制 复杂 链表 的 第 三 步 


TE: 把 第 二 步 得 到 的 链表 拆 分 成 两 个 链表 ， 奇 数位 置 上 的 结 点 组 成 原始 链表 ， 偶 数位 置 
上 的 结 点 组 成 复制 出 来 的 链表 。 


要 实现 第 三 步 的 操作 ， 也 不 是 很 难 的 事情 。 其 对 应 的 代码 如 下 : 





ComplexListNode* ReconnectNodes (ComplexListNode* pHead) 
{ 
ComplexListNode* pNode = pHead; 
ComplexListNode* pClonedHead = NULL; 
ComplexListNode* pClonedNode = NULL; 


if(pNode != NULL) 

{ 
pClonedHead = pClonedNode = pNode->m_pNext; 
pPNode->m_pNext = pClonedNode->m_pNext; 
PNode = pNode->m_pNext; 

} 


while (pNode != NULL) 

{ 
pClonedNode->m_pNext = pNode->m_pNext; 
pClonedNode = pClonedNode->m_pNext; 
pNode->m_pNext = pClonedNode->m_pNext; 
pNode = pNode->m_pNext; 

} 


return pClonedHead; 


我 们 把 上 面 三 步 合 起 来 ， 就 是 复制 链表 的 完整 过 程 : 


ComplexListNode* Clone (ComplexListNode* pHead) 
{ 
CloneNodes (pHead) ; 
ConnectSiblingNodes (pHead) ; 
return ReconnectNodes (pHead) ; 


} 
源 代码 : 
本 题 完整 的 源 代码 详 风 26_CopyComplexList 项 目 。 
测试 用 例 : 


e ”功能 测试 (包括 结 点 中 的 m_pSibling 指 向 结 点 自身 ， 两 个 结 点 
的 m_pSibling 形 成 环 状 结构 ， 链 表 中 只 有 一 个 结 点 )。 
e 特殊 输入 测试 (指向 链表 头 结 点 的 指针 为 NULL 指 针 〉。 


本 题 考点 : 





@ 考查 应 聘 者 对 复杂 问题 的 思维 能 力 。 本 题 中 的 复杂 链表 是 一 种 
不 太 常 见 的 数据 结构 ， 而 且 复 制 这 种 链表 的 过 程 也 较为 复杂 。 我 们 把 复 
杂 链 表 的 复制 过 程 分 解 成 三 个 步骤 ， 同 时 把 每 一 个 步骤 都 用 图 形 化 的 方 
式 表示 出 来 ， 这 些 方法 都 能 帮助 我 们 理 清 思路 。 写 代码 的 时 候 ， 我 们 为 
每 一 个 步骤 定义 一 个 子 函 数 ， 最 后 在 复制 函数 中 先后 调用 者 3 个 函数 。 
有 了 这 些 清晰 的 思路 之 后 再 写 代 码 ， 就 容易 多 了 。 

@ 考查 应 聘 者 分 析 时 间 效 率 和 空间 效率 的 能 力 。 当 应 聘 者 提出 第 
一 种 和 第 二 种 思路 的 时 候 ， 面 试 官 会 提示 此 时 在 效率 上 还 不 是 最 优 解 。 
A 0 

FEB? o 


面试 题 27: 二 又 搜索 树 与 双 问 链表 








题目 : 输入 一 柠 二 又 搜索 树 ， 将 该 二 又 搜索 树 转换 成 一 个 
排序 的 双 同 链表 。 要 求 不 能 创建 任何 新 的 结 点 ， 只 能 调整 树 中 
结 点 指针 的 指向 。 比 如 输入 图 4.12 中 左边 的 二 又 搜索 树 ， 则 输 
出 转换 之 后 的 排序 双 癌 链表。 





图 4.12 ”一 棵 二 又 搜索 树 及 转换 之 后 的 排序 双向 链表 
二 又 树 结 点 的 定义 如 下 : 


struct BinaryTreeNode 


{ 


int m nValue; 
BinaryTreeNode* m_pLeft; 
BinaryTreeNode* m_pRight; 


和 

在 二 义 树 中 ， 每 个 结 点 都 有 两 个 指 同 子 结 反 的 指针 。 在 双 问 链表 
中 ， 每 个 结 皮 也 有 两 个 指针 ， 它 们 分 列 指 问 前 一 个 结 点 和 后 一 个 结 反 。 
由 于 这 两 种 结 点 的 结构 相似 ， 同 时 二 又 搜索 树 也 是 一 种 排序 的 数据 结 


构 ， 因 此 在 理论 上 有 可 能 实现 二 又 搜索 树 和 排序 的 双 辐 链表 的 转换 。 在 
搜索 二 又 树 中 ， 左 子 结 点 的 值 总 是 小 于 父 结 点 的 值 ， 右 子 结 点 的 值 总 是 
大 于 父 结 点 的 值 。 因 此 我 们 在 转换 成 排序 双 同 链表 时 ， 原 先 指 癌 左 子 结 
点 的 指针 调整 为 链表 中 指向 前 一 个 结 点 的 指针 ， 原 先 指 向 右 子 结 点 的 指 
针 调 整 为 链表 中 指向 后 一 个 结 点 指针 。 接 下 来 我 们 考虑 该 如 何 转换 。 

由 于 要 求 转换 之 后 的 链表 是 排 好 序 的 ， 我 们 可 以 中 序 遍 历 树 中 的 每 
一 个 结 点 ， 这 是 因为 中 序 遍 历 算法 的 特点 是 按照 从 小 到 大 的 顺序 遍历 二 
叉 树 的 每 一 个 结 点 。 当 遍历 到 根 结 点 的 时 候 ， 我 们 把 树 看 成 三 部 分 : 值 
为 10 的 结 点 、 根 结 点 值 为 6 的 左 子 树 、 根 结 点 值 为 14 的 右 子 树 。 根 据 排 
序 链表 的 定义 ， 值 为 10 的 结 点 将 和 它 的 左 子 树 的 最 大 一 个 结 点 ( 即 值 为 
8 的 结 点 〉 链接 起 来 ， 同 时 它 还 将 和 右 子 树 最 小 的 结 点 ( 即 值 为 12 的 结 
点 ) 链接 起 来 ， 如 图 4.13 所 示 。 




















图 4.13 ”把 二 又 搜索 树 看 成 三 部 分 














TE: 根 结 点 、 左 子 树 和 石子 树 。 在 把 左 、 右 子 树 都 转换 成 排序 的 双向 链表 之 后 再 和 根 结 
点 链接 起 来 ， 整 棵 二 又 搜索 树 也 就 转换 成 了 排序 的 双 癌 链表 。 


按照 中 序 壳 历 的 顺序 ， 当 我 们 过 历 转换 到 根 结 点 〈 值 为 10 的 结 氮 ) 
时 ， 它 的 左 子 树 已 经 转换 成 一 个 排序 的 链表 了 ， 并 且 处 在 链表 中 的 最 后 
一 个 结 点 是 当前 值 最 大 的 结 点 。 我 们 把 值 为 8 的 结 点 和 根 结 点 链接 起 
来 ， 此 时 链表 中 的 最 后 一 个 结 点 融 是 10 了 。 接 独 我 们 去 过 历 转换 右 子 
树 ， 并 把 根 结 点 和 右 子 树 中 最 小 的 结 点 链接 起 来 。 至 于 怎么 去 转换 它 的 
oe ee eee 
以 用 递归 。 

基于 上 述 分 析 过 程 ， 我 们 可 以 写 出 如 下 代码 : 














BinaryTreeNode* Convert (BinaryTreeNode* pRootOfTree) 


{ 
BinaryTreeNode *pLastNodeInList = NULL; 
ConvertNode (pRootOfTree, &pLastNodeInList); 


// pLastNodeInList 指向 双向 链表 的 尾 结 点 ， 

// 我 们 需要 返回 头 结 点 

BinaryTreeNode *pHeadOfList = pLastNodeInList,; 

while (pHeadOfList != NULL && pHeadOfList->m_pLeft != NULL) 
pHeadOfList = pHeadOfList->m_pLeft; 


return pHeadOfList; 
} 


void ConvertNode (BinaryTreeNode* pNode, BinaryTreeNode** 
pLastNodeInList) 


{ 
if (pNode == NULL) 
return; 


BinaryTreeNode *pCurrent = pNode; 


if (pCurrent->m_pLeft != NULL) 
ConvertNode (pCurrent-—>m_pLeft, pLastNodeInList) ; 


pCurrent->m_pLeft = *pLastNodelInList; 
if (*pLastNodeInList != NULL) 
(*pLastNodeInList)->m_pRight = pCurrent; 


*pLastNodeInList = pCurrent; 


if (pCurrent->m_pRight != NULL) 
ConvertNode (pCurrent->m_pRight, pLastNodeInList); 


在 上 面 的 代码 中 ， 我 们 用 pLastNodeInList 指 向 已 经 转换 好 的 链表 的 
最 后 一 个 结 点 (也 是 值 最 大 的 结 点 ) 。 当 我 们 这 历 到 值 为 10 的 结 点 的 时 
候 ， 它 的 左 子 树 都 已 经 转换 好 了 ， 因 此 pLastNodeInList 指 向 值 为 8 的 结 
点 。 接 着 把 根 结 点 链接 到 链表 中 之 后 ， 值 为 10 的 结 点 成 了 链表 中 的 最 后 
一 个 结 点 (新 的 值 最 大 的 结 点 ) ， 于 是 pLastNodelInList 指 向 了 这 个 值 为 
10 的 结 点 。 接 下 来 把 pLastNodeInList ”作为 参数 传 入 函数 递归 遍历 右 子 
树 。 我 们 找到 右 子 树 中 最 左边 的 子 结 点 〈 值 为 12 的 结 点 ， 在 右 子 树 中 值 


最 小 ) ， 并 把 该 结 点 和 值 为 10 的 结 点 链接 起 来 。 
源 代码 : 
本 题 完整 的 源 代码 详 见 27_ConvertBinarySearchTree 项 目 。 
测试 用 例 : 


e 功能 汕 试 《输入 的 二 又 树 是 完全 二 又 树 ， 所 有 结 点 都 没有 左 / 右 
子 树 的 二 又 树 ， 只 有 一 个 结 点 的 二 叉 树 〉。 
。 特殊 输入 测试 ( 指 回 三 又 树 根 结 反 的 指针 为 NULL 指 针 〉。 


本 题 考点 : 


© 考查 应 聘 者 分 析 复 傈 问题 的 能 力 。 无 论 是 二 叉 树 还 是 双 癌 链 
表 ， 都 有 很 多 指针 。 要 实现 这 两 种 不 同 数 据 结 构 的 转换 ， 需 要 调整 大 量 
的 指针 ， 因 此 这 个 过 程 会 很 复杂 。 为 了 把 这 个 复杂 的 问题 分 析 清 楚 ， 我 
们 可 以 把 树 分 为 三 个 部 分 : 根 结 点 、 左 子 树 和 右 子 树 ， 然 后 把 左 子 树 中 
最 大 的 结 点 、 根 结 皮 、 右 子 树 中 最 小 的 结 点 链接 起 来 。 至 于 如 何 把 左 子 
树 和 右 子 树 内 部 的 结 反 链接 成 链表 ， 那 和 原来 的 问题 的 实质 是 一 样 的 ， 
因此 可 以 递归 人 解决。 解决 这 个 问题 的 关键 在 于 把 一 个 大 的 问题 分 解 成 几 
个 小 问题 ， 并 递归 地 解决 小 问题 。 

o 考 香 对 二 又 树 、 双 辐 链 表 的 理解 及 编程 能 


面试 题 28: 字符 串 的 排列 


题目 ;输入 一 个 字符 串 ， 打 印 出 该 字符 串 中 字符 的 所 有 排 
列 。 例 如 输入 字符 串 abc， 则 打印 出 由 字符 a、b、c 所 能 排列 出 
来 的 所 有 字符 串 abc、acb、bac、bca、cab 和 cba。 

如 何 求 出 几 个 字符 的 所 有 排列 ， 很 多 人 都 不 能 一 下 子 想 出 解决 方 
案 。 那 我 们 是 不 是 可 以 考虑 把 这 个 复杂 的 问题 分 解 成 小 的 问题 呢 ? 比 
如 ， 我 们 把 一 个 字符 串 看 成 由 两 部 分 组 成 : 第 一 部 分 为 它 的 第 一 个 字 
符 ， 第 二 部 分 是 后 面 的 所 有 字符 。 在 图 4.14 中 ， 我 们 用 两 种 不 同 的 背景 
颜色 区 分 字符 串 的 两 部 分 。 

我 们 求 整个 字符 串 的 排列 ， 可 以 看 成 两 步 : 首先 求 所 有 可 能 出 现在 
第 一 个 位 置 的 字符 ， 即 把 第 一 个 字符 和 后 面 所 有 的 字符 交换 。 图 4.14 就 
是 分 别 把 第 一 个 字符 a 和 后 面 的 b、c 等 字符 交换 的 情形 。 首 先 固定 第 一 
个 字符 (如 图 4.14 (a) 所 示 ) ， 求 后 面 所 有 字符 的 排列 。 这 个 时 候 我 们 
仍 把 后 面 的 所 有 字符 分 成 两 部 分 : 后 面 字 符 的 第 一 个 字符 ， 以 及 这 个 字 














符 之 后 的 所 有 字符 。 然 后 把 第 一 个 字符 逐一 和 它 后 面 的 字符 交换 《〈 如 图 
4.14 (b) 所 示 )...... 


图 4.14 求 字符 捉 的 排列 的 过 程 
VE: (a) 把 字符 串 分 为 两 部 分 ， 一 部 分 是 字符 串 的 第 一 个 字符 ， 另 一 部 分 是 第 一 个 字符 
以 后 的 所 有 字符 〈 有 阴影 背景 的 区 域 ) 。 接 下 来 我 们 求 阴影 部 分 的 字符 串 的 排列 。 b) 拿 第 一 
个 字符 和 它 后 面 的 字符 逐个 交换 。 
分 析 到 这 里 ， 我 们 就 可 以 看 出 ， 这 其 实 是 很 典型 的 递归 思路 ， 于 是 
我 们 不 难 写 出 如 下 代码 : 























void Permutation(char* pStr) 


Permutation(pStr, pStr); 


} 


{ 
for(char* pCh = pBegin; *pCh != '\0'; ++ pCh) 
{ 
h; 


Permutation(pStr, pBegin + 1); 


temp = Ch; 
kpCh = *pBegin; 
*pBegin = temp; 


} 
} 

在 函数 Permnutation (char pStr, char’ pBegin) 中 ， 指 针 pStr 指 向 整个 
字符 串 的 第 一 个 字符 ，pBegin 指 同 当前 我 们 做 排列 操作 的 字符 串 的 第 一 
个 字符 。 在 每 一 次 递归 的 时 候 ， 我 们 从 pBegin 辐 后 扫描 每 一 个 字符 〈 即 
指针 pCh 指 向 的 字符 〉。 在 交换 pBegin 和 pCh 指 向 的 字符 之 后 ， 我 们 再 
对 pBegin 后 面 的 字符 串 递 归 地 做 排列 操作 ， 直 至 pPBegin 指 回 字 符 串 的 末 
尾 。 

源 代码 : 

本 题 完整 的 源 代码 详 见 28_StringPermutation 项 目 。 


测试 用 例 : 





o 功能 测试 《输入 的 字符 串 中 有 1 个 或 者 多 个 字符 ) 。 
。 特殊 输入 测试 (输入 的 字符 串 的 内 容 为 空 或 者 是 NULL 指 
2 


本 题 考点 : 


。 考查 思维 能 力 。 当 整个 问题 看 起 来 不 能 直接 解决 的 时 候 ， 应 聘 
者 能 否 想到 把 字符 串 分 成 两 部 分 ， 从 而 把 大 问题 分 解 成 小 问题 来 解决 ， 
征 能 人 否 顺利 解决 这 个 问题 的 关键 。 

e 考查 对 递归 的 理解 和 编程 能 力 。 


本 题 扩展 : 


如 果 不 是 求 字 符 的 所 有 排列 ， 而 是 求 字 符 的 所 有 组 合 ， 应 该 怎么 办 
呢 ? 还 是 输入 三 个 字符 8g、b、c， 则 它们 的 组 合 有 a、b、c、ab、ac、 
bc、abc。 当 交换 字符 串 中 的 两 个 字符 时 ， 虽 然 能 得 到 两 个 不 同 的 排 
列 ， 但 却 是 同一 个 组 合 。 比 如 ab 和 ba 是 不 同 的 排列 ， 但 只 算 一 个 组 合 。 

如 果 输 入 n 个 字符 ， 则 这 mn 个 字符 能 构成 长 度 为 1 的 组 合 、 长 度 为 2 的 
rk dusts 、 长 度 为 n 的 组 合 。 在 求 n 个 字符 的 长 度 为 mn (1<m<n) 的 组 
合 的 时 候 ， 我 们 把 这 n 个 字符 分 成 两 部 分 : 第 一 个 字符 和 其 余 的 所 有 字 
符 。 如 果 组 合 里 包含 第 一 个 字符 ， 则 下 一 步 在 剩余 的 字符 里 选取 m 一 1 
个 字符 ;， 如 果 组 合 里 不 包含 第 一 个 字符 ， 则 下 一 步 在 剩余 的 n 一 1 个 字符 
里 选取 m 个 字符 。 也 就 是 说 ， 我 们 可 以 把 求 n 个 字符 组 成 长 度 为 m 的 组 合 
的 问题 分 解 成 两 个 子 问 题 ， 分 别 求 n 一 1 个 字符 串 中 长 度 为 m 一 1 的 组 
合 ， 以 及 求 n 一 1 个 字符 的 长 度 为 m 的 组 合 。 这 两 个 子 问题 都 可 以 用 递归 
的 方式 解决 。 

相关 题目 : 

1. 输入 一 个 含有 8 个 数字 的 数组 ， 判 断 有 没有 可 能 把 这 8 个 数字 分 
别 放 到 正方 体 的 8 个 顶点 上 《如 图 4.15 所 示 ) ， 使 得 正方 体 上 三 组 相对 
的 面 上 的 4 个 顶点 的 和 都 相等 。 








a2 


a6 











图 4.15 ”把 8 个 数字 放 到 正方 体 的 8 个 顶点 上 


这 相当 于 先 得 到 al、a2、a3、a4、a5、a6、a7 和 a8 这 8 个 数字 的 所 有 
排列 ， 然 后 判断 有 没有 某 一 个 的 排列 符合 题目 给 定 的 条 件 ， 即 al 十 a2 十 
a3 十 a4 二 a5 十 a6 十 a7 十 a8，al 十 a3 十 a5 十 a7 二 a2 十 a4 十 a6 十 a88， 并 且 al 十 
a2 十 a5 十 a6 二 a3 十 a4 十 a7 十 a8。 

2. 在 8x8 的 国际 象棋 上 摆 放 8 个 旺 后 ， 使 其 不 能 相互 攻击 ， 即 任意 
两 个 星 后 不 得 处 在 同一 行 、 同 一 列 或 者 同一 对 角 线 上 。 图 4.16 中 的 每 个 
黑色 格子 表示 一 个 皇后 ， 这 束 是 一 种 符合 条 件 的 摆 放 方法 。 请 问 总 共有 
多 少 种 符合 条 件 的 摆 法 ? 



































图 4.16 ”8x8 的 国际 象棋 棋盘 上 捍 着 8 个 皇后 ( 





色 小 方 格 )， 任 意 两 个 皇后 不 在 同一 行 、 同 一 


We 








列 或 者 同一 对 角 线 上 

由 于 8 个 星 后 的 任意 两 个 不 能 处 在 同一 行 ， 那 么 肯定 是 每 一 个 星 后 
占据 一 行 。 于 是 我 们 可 以 定义 一 个 数组 ColumnIndex[8]， 数 组 中 第 i 个 数 
字 表 示 位 于 第 i 行 的 星 后 的 列 写 。 先 把 数组 ColumnIndex 的 8 个 数字 分 别 
用 0 一 7 初始 化 ， 接 下 来 就 是 对 数组 ColumnIndex 做 全 排列 。 因 为 我 们 是 
用 不 同 的 数字 初始 化 数组 ， 所 以 任意 两 个 蛙 后 肯定 不 同 列 。 我 们 只 需 判 
断 每 一 个 排列 对 应 的 8 个 星 后 是 不 是 在 同一 对 角 线 上 ， 也 束 是 对 于 数组 
的 两 个 下 标 i 和 j， 是 不 是 i-j=ColumnIndex[i]-ColumnIndex[j] 或 者 j- 
i=ColumnIndex{[i]-ColumnIndex{[j]. 

举一反三 : 

如 果 面 试题 是 按照 一 定 要 求 摆 放 若 干 个 数字 ， 我 们 可 以 先 求 出 这 些 数字 的 所 有 排列 ， 然 
后 再 判断 每 个 排列 是 不 是 满足 题目 给 定 的 要 求 。 


45 本章 小 结 


面试 的 时 候 我 们 难免 会 遇 到 难题 ， 画 图 、 举 例子 和 分 解 这 三 种 办 法 
能 够 帮助 我 们 解决 复杂 的 问题 (如 图 4.17 所 示 》。 
















































































图 4.17 ”解决 复杂 问题 的 三 种 方法 ; 画图 、 举 例子 和 分 解 。 

图 形 能 使 抽象 的 问题 形象 化 。 当 面试 题 涉及 链表 、 二 又 树 等 数据 结 
构 时 ， 如 果 在 纸 上 男 几 张 草图 ， 题 目 中 隐藏 的 规律 就 有 可 能 变 得 很 直 
观 。 








一 两 个 例子 能 使 抽象 的 问题 具体 化 。 很 多 与 算法 相关 的 问题 都 很 抽 
象 ， 未 必 一 眼 就 能 看 出 它们 的 规律 。 这 个 时 候 我 们 不 妨 举 几 个 例子 ， 一 
人 
题 的 穷 门 。 

把 复杂 问题 分 解 成 各 干 个 小 问题 ， 是 解决 很 多 复杂 问题 的 有 效 方 
法 。 如 果 我 们 过 到 的 问题 很 大 ， 可 以 尝试 先 把 大 问题 分 解 成 小 问题 ， 然 
后 再 递归 地 解决 这 些小 问题 。 分 治 法 、 动 态 规划 等 方法 都 是 应 用 分 解 复 
杂 问 题 的 思路 。 





第 5 章 “优化 时 间 和 空间 效率 
5.1 面试 官 谈 效率 


“通常 针对 一 些 senior dev 的 candidates 会 问 一 些 关 于 时 间 、 衬 间 效 率 的 问题 ， 这 能 够 体现 一 
个 应 聘 者 较 好 的 编程 素质 和 能 力 。” 





























— X] (Autodesk, $F FEI) 
“面试 时 一 般 会 直接 要 求 空间 和 时 间 复 杂 度 ， 这 两 者 都 很 重要 。” 
— 张 表 〈 百 度 ， 高 级 软件 工程 师 ) 


“我 们 有 很 多 考查 时 间 、 空 间 效率 这 方面 的 问题 。 通 常 两 者 都 给 应 聘 者 限定 ， 然 后 让 他 给 
出 解决 方案 。” 

































































— ik RA AE, ARAH) 


=e “只 要 不 是 特别 大 的 内 存 开销 ， 时 间 复 杂 度 比较 重要 。 因 为 改进 时 间 复 杂 度 对 算法 的 要 求 
iyo ” 
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一 一 吴斌 (NVidia, Graphics Architect) 

“空间 换 时 间 还 是 时 间 换 空间 ， 这 要 看 具体 的 题目 了 。 对 于 普通 的 应 用 ， 一 般 是 空间 换 时 

间 ， 因 为 通常 用 户 更 关心 速度 ， 而 且 一 般 有 足够 的 存储 空间 允许 这 么 做 。 但 对 于 现在 的 一 般 嵌 
入 式 设 备 ， 很 多 时 候 空间 换 时 间 就 不 现实 了 ， 因 为 存储 空间 太 少 了 。” 
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陈 歼 明 (微软 ，SDE ID) 





5.2 ”时 间 效 率 


由 于 每 个 人 都 希望 软件 的 啊 应 时 间 尽 量 短 一 些 ， 所 以 软件 公司 都 很 
重视 软件 的 时 间 性 能 ， 都 会 在 发 布 软件 之 前 花 不 少 精力 做 时 间 效 率 优 
化 。 这 也 就 不 难 理解 为 什么 很 多 公司 的 面试 官 都 把 代码 的 时 间 效 率 当 做 
一 个 考 香 重点 。 面 试 官 除 了 考 碍 应 聘 者 的 编程 能 力 之 外 ， 还 关注 应 聘 者 
有 没有 不 断 优化 效率 、 奶 求 完 美的 态度 和 能 力 。 

首先 ， 我 们 的 编程 习惯 对 代码 的 时 间 效率 有 很 大 影响 。 比 如 
C/C++ 程 序 员 要 养 成 采用 引用 《或 指针 ) 传递 复杂 类 型 参数 的 习惯 。 如 
条 采用 值 传递 的 方式 ， 从 形 参 到 实 参 会 产生 一 次 复制 操作 。 这 样 的 复制 
是 多 余 的 操作 ， 我 们 应 该 尽量 避免 。 再 举 个 例子 ， 如 果 用 C# 做 多 次 字符 
串 的 拼接 操作 ， 不 要 多 次 用 String 的 + 运算 符 来 拼接 字符 串 ， 因 为 这 样 会 
产生 很 多 String 的 临时 实例 ， 造 成 时 间 和 空间 的 浪费 。 更 好 的 办 法 是 用 


























StringBuilder 的 Append 方 法 来 完成 字符 串 的 拼接 。 如 果 我 们 平时 不 太 注 
意 这 些 影响 代码 效率 的 细节 ， 没 有 养 成 好 的 编码 习惯 ， 那 么 我 们 的 代码 
可 能 就 会 让 面试 官 大 失 所 望 。 

其 次 ， 即 使 同一 个 算法 用 循环 和 递归 两 种 思路 实现 的 时 间 效 率 可 能 
会 大 不 一 样 。 递 归 的 本 质 是 把 一 个 大 的 复杂 问题 分 解 成 两 个 或 者 多 个 小 
的 简单 的 问题 。 如 果 小 问题 中 有 相互 重 肘 的 部 分 ， 那 么 直接 用 递归 实现 
虽然 代码 显得 很 简洁 ， 但 时 间 效 率 可 能 会 非常 差 ( 详 细 讨 论 见 本 书 2.4.2 
W) 。 对 于 这 种 类 型 的 题目 ， 我 们 可 以 用 递归 的 思路 来 分 析 问 题 ， 但 写 
代码 的 时 候 可 以 用 数组 (一 维 或 者 多 维 数 组 ) 来 保存 中 间 结 果 基 于 循环 
a a aa R 


FR, ARAB HIET Ti] RCE IE RE SEL IES OT BH A AI, TH JER A) E 
握 程度 。 同 样 是 查找 ， 如 果 是 顺序 查找 需要 O(n) 的 时 间 ; 如 果 输 入 
的 是 排序 的 数组 则 只 需要 O (logn)〉 的 时 间 ; 如 果 事 先 已 经 构造 好 了 哈 
ER, MAREO D 时 间 就 能 完成 。 我 们 只 有 对 第 见 的 数据 结构 和 
BU T RTH, REE SS UIT REPE E K E SS RR 
问题 。 

最 后 ， 应 聘 者 在 面试 的 时 候 要 展示 敏捷 的 轧 维 能 力 和 人 退 求 完美 的 激 
情 。 听 到 题目 的 时 候 ， 我 们 一 般 很 快 就 能 想到 最 直观 的 算法 。 这 个 最 直 
观 的 办 法 很 有 可 能 不 是 最 优 的 ， 但 也 不 妨 在 第 一 时 间 告 诉 面试 官 ， 这 样 
面试 官 至 少 会 觉得 我 们 思维 比较 敏捷 。 我 们 想到 儿 种 思路 之 后 面试 官 可 
能 仍然 不 满意 ， 还 在 提示 我 们 有 更 好 的 办 法 。 这 个 时 候 我 们 一 定 不 能 轻 
言 放 径 ， 而 要 表现 出 积极 思考 的 态度 ， 努 力 从 不 同 的 角度 去 思考 问题 。 
有 些 题目 很 难 ， 面 试 官 甚至 不 期 符 应 聘 者 在 短 短 儿 十 分 钟 里 想 出 完美 的 
解法 ， 但 他 会 希望 应 聘 者 能 够 有 激情 、 有 耐心 去 符 试 新 的 思路 ， 而 不 是 
倍 到 难题 束 退 缩 。 在 面试 的 时 候 ， 应 聘 者 的 态度 和 诅 情 对 最 终 的 面试 络 
果 也 有 很 重要 的 影响 。 


面试 题 29: 数组 中 出 现 次 数 超过 一 半 的 数字 


题目 : 数组 中 有 一 个 数字 出 现 的 次 数 超过 数组 长 度 的 一 
半 ， 请 找 出 这 个 数字 。 例 如 输入 一 个 长 度 为 9 的 数组 
{1,2,3,2,2,2,5,4,2}。 由 于 数字 2 在 数组 中 出 现 了 5 次 ， 超 过 数组 
长 度 的 一 半 ， 因 此 输出 2。 

看 到 这 道 题 很 多 应 聘 者 就 会 想 要 是 这 个 数组 是 排序 的 数组 就 好 了 。 
如 果 是 排 好 序 的 数组 ， 那 么 我 们 就 能 很 容易 统计 出 每 个 数字 出 现 的 次 



































数 。 题 目 给 出 的 数组 没有 说 是 排序 的 ， 因 此 我 们 需要 先 给 它 排 序 。 排 序 
的 时 间 复 杂 度 是 O Cnlogn) 。 最 直观 的 算法 通常 不 是 面试 官 满意 的 算 
法 ， 接 下 来 我 们 试 着 找 出 更 快 的 算法 。 


解法 一 :基于 Partition 函数 的 O (n) 算法 


如 果 我 们 回 到 题目 本 里 仔细 分 析 ， 束 会 及 现 前 面 的 思路 并 没有 考虑 
到 数组 的 特性 : 数组 中 有 一 个 数字 出 现 的 次 数 超过 了 数组 长 度 的 一 半 。 
如 傈 把 这 个 数组 排序 ， 那 么 排序 之 后 位 于 数组 中 间 的 数字 一 定 就 是 那个 
出 现 次 数 超过 数组 长 度 一 半 的 数字 。 也 就 是 说 ， 这 个 数字 就 是 统计 学 上 
的 中 位 数 ， 即 长 度 为 n 的 数组 中 第 n/2 大 的 数字 。 我 们 有 成 熟 的 O Cn) 的 
算法 得 到 数组 中 任意 第 k 大 的 数字 。 

这 种 算法 是 受 快速 排序 算法 的 局 发 。 在 随机 快速 排序 算法 中 ， 我 们 
先 在 数组 中 随机 选择 一 个 数字 ， 然 后 调整 数组 中 数字 的 顺序 ， 使 得 比 选 
中 的 数字 小 数字 都 排 在 它 的 左边 ， 比 选中 的 数字 大 的 数字 都 排 在 它 的 右 
边 。 如 采 这 个 选中 的 数字 的 下 标 刚 好 是 n/2， 那 么 这 个 数字 就 是 数组 的 
中 位 数 。 如 果 它 的 下 标 大 于 m2， 那 么 中 位 数 应 该 位 于 它 的 左边 ， 我 们 
可 以 接着 在 它 的 左边 部 分 的 数组 中 查找 。 如 果 它 的 下 标 小 于 nm/2， 那 么 
中 位 数 应 该 位 于 它 的 右边 ， 我 们 可 以 接着 在 它 的 右边 部 分 的 数组 中 得 
找 。 这 是 一 个 典型 的 递归 过 程 ， 可 以 用 如 下 代码 实现 : 





























int MoreThanHalfNum(int* numbers, 


int length) 
{ 


if (CheckInvalidArray (numbers, length) ) 
return 0; 


int middle = length >> 1; 

int start = 0; 

int end = length - 1; 

int index = Partition(numbers, length, 
while (index != middle) 

| 


start, end); 


if (index > middle) 


{ 


end = index 一 1; 
index = Partition(numbers, length, start, end); 
} 
else 
{ 
start = index + 1; 
index = Partition(numbers, length, start, end); 
} 
} 
int result = numbers [middle]; 
if (!CheckMoreThanHalf (numbers, length, result) ) 
result = 0; 


return result; 
} 

上 述 代 码 中 的 函数 Partition 是 完成 快速 排序 的 基础 。 我 们 在 本 书 的 
2.4.1 节 详细 讨论 了 这 个 函数 ， 这 里 不 再 重复 。 

在 面试 的 时 候 ， 除 了 要 完成 基本 功能 即 找到 符合 要 求 的 数字 之 外 ， 
还 要 考虑 一 些 无 效 的 输入 。 如 果 函 数 的 输入 参数 是 一 个 指针 〈 数 组 在 参 
数 传 递 的 时 候 退 化 为 指针 ) ， 就 要 考虑 这 个 指针 可 能 为 NULL。 下 面 的 
函数 CheckInvalidArray 用 来 判断 输入 的 数组 是 不 是 无 效 的 。 题 目 中 说 数 
组 中 有 一 个 数字 出 现 次 数 超过 数组 长 度 的 一 半 ， 如 果 输 入 的 数组 中 出 现 
频率 最 高 的 数字 都 没有 达到 这 个 标准 那 该 怎么 办 ? 这 就 是 我 们 定义 了 一 
个 CheckMoreThanHalf 函 数 的 原因 。 面 试 的 时 候 我 们 要 全 面 考虑 这 些 情 
况 ， 才 能 让 面试 官 完 全 满意 。 下 面 的 代码 用 一 个 全 局 变量 来 表示 输入 无 
效 的 情况 。 更 多 关于 出 错 处 理 的 讨论 ， 详 见 本 书 3.3 节 。 











bool g bIinputIinvalid = false; 


ool CheckInvalidArray(int* numbers, int length) 


mo 


g bInputinvalid = false; 
if (numbers == NULL && length <= 0) 
g bInputInvalid = true; 


return g bInputiInvalid; 


bool CheckMoreThanHalf(int* numbers, int length, int number) 


M 。 
J; 


0; i < length; ++i) 


int times 
forlint i 


if(numbers[i] == number) 


bool isMoreThanHalf = true; 
if(times * 2 <= length) 
{ 
g bInputiInvalid = true; 
isMoreThanHalf = false; 


} 


return isMoreThanHalf; 


解法 二 : ARR AKO Cn) 的 算法 


接 下 来 我 们 从 另外 一 个 角度 来 解决 这 个 问题 。 数 组 中 有 一 个 数字 出 
现 的 次 数 超过 数组 长 上 度 的 一 半 ， 也 就 是 说 它 出 现 的 次 数 比 其 他 所 有 数字 
出 现 次 数 的 和 还 要 多 。 因 此 我 们 可 以 考虑 在 衣 历 数组 的 时 候 保 存 两 个 
值 : 一 个 是 数组 中 的 一 个 数字 ， 一 个 是 次 数 。 当 我 们 过 有 历 到 下 一 个 数字 
的 时 候 ， 如 果 下 一 个 数字 和 我 们 之 前 保存 的 数字 相同 ， 则 次 数 加 1; 如 
本 下 一 个 数字 和 我 们 之 前 保存 的 数字 不同 ， 则 次 数 减 1。 如 胰 次 数 为 
零 ， 我 们 需要 保存 下 一 个 数字 ， 并 把 次 数 设 为 1。 由 于 我 们 要 找 的 数字 
出 现 的 次 数 比 其 他 所 有 数字 出 现 的 次 数 之 和 还 要 多 ， 那 么 要 找 的 数字 肯 
定 是 最 后 一 次 把 次 数 设 为 1 时 对 应 的 数字 。 


下 面 是 这 种 思路 的 参考 代码 : 


int MoreThanHalfNum(int* numbers, int length) 


{ 
if (CheckInvalidArray (numbers, length) ) 
return 0; 
int result = numbers[0]; 
int times = 1 
for(int i = 1; i < length; ++i) 
{ 
if (times == 0) 
{ 
result = numbers[i 
times = 1 
} 
else if(numbers[i] == result) 
times++; 
times—— 
} 
if(!CheckMoreThanHalf (numbers, length, result)) 
result = 0; 
return result 
} 


和 第 一 种 思路 一 样 ， 我 们 也 要 检验 输入 的 数组 是 不 是 有 效 的 ， 这 里 
不 再 重复 。 


解法 比较 


上 述 两 种 算法 的 时 间 复 杂 度 都 是 O Cn) 。 基 于 Partition 的 算法 的 时 
间 复 杂 度 的 分 析 不 是 很 直观 ， 本 书 限于 篇 幅 不 作 详 细 讨 论 ， 感 兴趣 的 读 
者 可 以 参考 《算法 导论 》 等 书籍 的 相关 章节 。 我 们 注意 到 在 第 一 个 解法 
中 ， 需 要 交换 数组 中 数字 的 顺序 ， 这 就 会 修改 输入 的 数组 。 我 们 是 不 是 
可 以 修改 输入 的 数组 呢 ? 在 面试 的 时 候 ， 我 们 可 以 和 面试 官 讨 论 ， 让 他 
如 果 面 试 官 说 不 能 修改 输入 的 数组 ， 那 束 只 能 采用 第 二 种 得 
大 下。 

源 代 码 : 

本 题 完整 的 源 代 人 码 详 见 29_MoreThanHalfNumber 项 目 。 














测试 用 例 : 


o 功能 测试 (输入 的 数组 中 存在 一 个 出 现 次 数 超 过 数组 长 度 一 半 
的 数字 ， 输 入 的 数组 中 不 存在 一 个 出 现 次 数 超过 数组 长 度 一 半 的 数 
T) 


ae 特殊 输入 测试 (输入 的 数组 中 只 有 一 个 数字 、 输 入 NULL 指 
) 。 

本 题 考点 : 

。 考查 对 时 间 复杂 度 的 理解 。 应 聘 者 每 想 出 一 种 解法 ， 面 试 官 都 
期 待 他 能 分 析出 这 种 解法 的 时 间 复 杂 度 是 多 少 。 

。 考查 思维 的 全 面 性 。 面 试 官 除了 要 求 应 聘 者 能 对 有 效 的 输入 返 
回 正确 的 结果 之 外 ， 同 时 也 期 待 应 聘 者 能 对 无 效 的 输入 作 相 应 的 处 理 。 


面试 题 30: 最 小 的 k 个 数 


题目 : 输入 n 个 整数 ， 找 出 其 中 最 小 的 k 个 数 。 例 如 输入 
4、5、1、6、2、7、3、8 这 8 个 数字 ， 则 最 小 的 4 个 数字 是 1、 
De ds 

这 道 题 最 简单 的 思路 葛 过 于 把 输入 的 n 个 整数 排序 ， 排 序 之 后 位 于 
最 前 面 的 k 个 数 束 是 最 小 的 k 个 数 。 这 种 思路 的 时 间 复 杂 上 度 是 
O Cnlogn) ， 面 试 官 会 提示 我 们 还 有 更 快 的 算法 。 


解法 一 : O(n) 的 算法 ， 只 有 当 我 们 可 以 修改 输入 的 数组 时 
可 用 


从 解决 面试 题 29“ 数 组 中 出 现 次 数 超过 一 半 的 数字 ”得 到 了 局 发 ， 我 
们 同样 可 以 基于 Partition 函 数 来 解决 这 个 问题 。 如 果 基 于 数组 的 第 k 个 数 
字 来 调整 ， 使 得 比 第 k 个 数字 小 的 所 有 数字 都 位 于 数组 的 左边 ， 比 第 k 个 
数字 大 的 所 有 数字 都 位 于 数组 的 右边 。 这 样 调整 之 后 ， 位 于 数组 中 左边 
的 k 个 数字 就 是 最 小 的 k 个 数字 (这 k 个 数字 不 一 定 是 排序 的 ) 。 下 面 是 
基于 这 种 思路 的 参考 代码 : 




















void GetLeastNumbers(int* input, int n, int* output, int k) 
{ 


if (input == NULL output == NULL k>n || n <= 0 k <= 0) 
return 

int start = 0; 

int end =n - 1; 

int index = Partition(input, n, start, end) 

while(index != k - 1) 

{ 


if(index > k - 1) 


{ 

end = index - 1; 

index = Partition(input, n, start, end); 
} 
{ 

start index + 1; 


index = Partition(input, n, start, end); 
} 
} 


For (int 2 = OF T < Ke Aki) 
output[i] = input[i]; 


采用 这 种 思路 是 有 限制 的 。 我 们 需要 修改 输入 的 数组 ， 因 为 函数 
Partition 会 调整 数组 中 数字 的 顺序 。 如 末 面 试 官 要 求 不 能 修改 输入 的 数 
组 ， 我 们 该 上 怎么 办 呢 ? 


解法 二 : O Cnlogk) 的 算法 ， 特 别 适合 处 理 海量 数据 


我 们 可 以 移 创 建 一 个 大 小 为 k 的 数据 容器 来 存储 最 小 的 k 个 数字 ， 接 
下 来 我 们 每 次 从 输入 的 n 个 整数 中 读 入 一 个 数 。 如 有 果 容 圳 中 已 有 的 数字 
少 于 k 个 ， 则 直接 把 这 次 读 入 的 整数 放 入 容 占 之 中 ; 如 果 容 器 中 己 有 k 个 
数字 了 ， 也 就 是 容器 已 满 ， 此 时 我 们 不 能 再 插入 新 的 数字 而 只 能 丛 换 已 
有 的 数字 。 找 出 这 已 有 的 k 个 数 中 的 最 大 值 ， 然 后 拿 这 次 待 插入 的 整数 
和 最 大 值 进 行 比较 。 如 果 竺 插入 的 值 比 当前 已 有 的 最 大 值 小 ， 则 用 这 个 
数 将 换 当 前 已 有 的 最 大 值 ;， 如 末 竺 插入 的 值 比 当前 已 有 的 最 大 值 还 妥 
i 

因此 当 容 器 满 了 之 后 ， 我 们 要 做 3 件 事情 : 一 是 在 k 个 整数 中 找到 最 


} 























大 数 ， 二 是 有 可 能 在 这 个 容器 中 删除 最 大 数 ， 三 是 有 可 能 要 插入 一 个 新 
的 数字 。 如 果 用 一 个 二 又 树 来 实现 这 个 数据 容器 ， 那 么 我 们 能 在 
O Clogk) 时 间 内 实现 这 三 步 操作 。 因 此 对 于 n 个 输入 数字 而 言 ， 总 的 时 
间 效 率 就 是 O Cnlogk) 。 

我 们 可 以 选择 用 不 同 的 二 又 树 来 实现 这 个 数据 容 妖 。 由 于 每 次 都 需 
要 找到 k 个 整数 中 的 最 大 数字 ， 我 们 很 容易 想到 用 最 大 堆 。 在 最 大 堆 
中 ， 根 结 点 的 值 总 是 大 于 它 的 子 树 中 任意 结 点 的 值 。 于 是 我 们 每 次 可 以 
EO (1) 得 到 已 有 的 k 个 数字 中 的 最 大 值 ， 但 需要 O Clogk) 时间 完成 
删除 及 插入 操作 。 

我 们 自己 从 头 实现 一 个 最 大 堆 需 要 一 定 的 代码 ， 这 在 面试 短 短 的 几 
十 分 钟 内 很 难 完成 。 我 们 还 可 以 采用 红 黑 树 来 实现 我 们 的 容器 。 红 黑 树 
通过 把 结 点 分 为 红 、 黑 两 种 颜色 并 根据 一 些 规则 确保 树 在 一 定 程度 上 是 
平衡 的 ， 从 而 保证 在 红 黑 树 中 人 查找、 删除 和 插入 操作 都 只 需要 
O (ogk) 时 间 。 在 STL 中 set 和 multiset 都 是 基于 红 黑 树 实现 的 。 如 果 面 
试 官 不 反对 我 们 用 STL 中 的 数据 容器 ， 我 们 就 可 以 直接 拿 过 来 用 。 下 面 
是 基于 STL 中 的 multiset 的 参考 代码 : 














typedef multiset<int, greater<int> > intSet; 
typedef multiset<int, greater<int> >::iterator setIterator; 


void GetLeastNumbers (const vector<int>& data, intSeté leastNumbers, 
int k) 
{ 

leastNumbers.clear(); 

ifik < 1 || data.size() < k) 

return; 

vector<int>::const_iterator iter = data.begin(); 

for(; iter =. data.end(); ++ iter) 

{ 


if ( (leastNumbers.size(} k 
leastNumbers.insert (*iter); 


jeca 
SLioet 


setiterator iterGreatest = leastNumbers.begin(); 


if(*iter < *(leastNumbers.begin())) 


{ 
leastNumbers.erase (iterGreatest); 
leastNumbers.insert(*iter); 

} 


解法 比较 


基于 函数 Partitiaon 的 第 一 种 解法 的 平均 时 间 复 杂 度 是 O Cn) ， 比 第 
二 种 思路 要 快 ， 但 同时 它 也 有 明显 的 限制 ， 比 如 会 修改 输入 的 数组 。 

第 二 种 解法 虽然 要 慢 一 点 ， 但 它 有 两 个 明显 的 优点 。 一 是 没有 修改 
输入 的 数据 (代码 中 的 变量 data，〉。 我 们 每 次 只 是 从 data 中 读 入 数字 ， 
所 有 的 写 操作 都 是 在 容器 leastNumbers 中 进行 的 。 二 是 该 算法 适合 海量 
数据 的 输入 《包括 百 度 在 内 的 多 家 公司 非常 喜欢 与 海量 输入 数据 相关 的 
问题 )》。 假 设 题目 是 要 求 从 海量 的 数据 中 找 出 最 小 的 k 个 数字 ， 由 于 内 
存 的 大 小 是 有 限 的 ， 有 可 能 不 能 把 这 些 海量 的 数据 一 次 性 全 部 载 入 内 
存 。 这 个 时 候 ， 我 们 可 以 从 辅助 存储 空间 (比如 硬盘 ) 中 每 次 读 入 一 个 
数字 ， 根 据 GetLeastNumbers 的 方式 判断 是 不 是 需要 放 入 容器 
leastNumbers 即 可 。 这 种 思路 只 要 求 内 存 能 够 容纳 leastNumbers 即 可 ， 因 























此 它 最 适合 的 情形 束 是 n 很 大 并 且 k 较 小 的 问题 。 
我 们 可 以 用 表 5.1 总 结 这 两 种 解法 的 特点 。 
表 5.1 两 种 算法 的 特点 比较 


基于 Partition 函 数 的 思路 基于 堆 或 者 红 黑 树 的 思路 


由 于 这 两 种 算法 各 有 优 缺 点 ， 各 目 适 用 于 不 同 的 场合 ， 因 此 应 聘 者 
在 动手 做 题 之 前 移 要 问 清 楚 题 目的 要 求 ， 包 括 输入 的 数据 量 有 多 大 、 能 
否 一 次 性 载 入 内 存 、 和 是 否 人 允许 交 换 输入 数据 中 数字 的 顺序 等 。 

面试 小 提示 : 


如 果 面 试 时 遇 到 的 面试 题 有 多 种 解法 ， 并 且 每 个 解法 都 各 有 优 缺 点 ， 那 么 我 们 要 向 面试 
启 问 清楚 题目 的 要 求 ， 输 入 的 特点 ， 从 而 选择 最 合适 的 解法 。 



















































































源 代码 : 
本 题 完整 的 源 代码 详 见 30_KLeastNumbers 项 目 。 
测试 用 例 : 


i 于 能 测试 《输入 的 数组 中 有 相同 的 数字 ， 输 入 的 数组 中 没有 间 
司 的 数字 ) 。 

。 边界 值 测试 “输入 的 k 等 于 1 或 者 等 于 数组 的 长 度 ) 

。 特殊 输入 测试 《k 小 于 1、k 大 于 数组 的 长 度 、 指 向 数组 的 指针 为 
NULL) 。 


本 题 考点 : 


e 考查 对 时 间 复 杂 度 的 分 析 能 力 。 面 试 的 时 候 每 想 出 一 个 解法 ， 
我 们 都 要 能 分 析出 这 种 解法 的 时 间 复 杂 上 度 是 多 少 。 

e ”如果 采用 第 一 种 思路 ， 本 题 考 人 对 Partition 函 数 的 理解 。 这 个 函 
数 既 是 快速 排序 的 基础 ， 也 可 以 用 来 查找 n 个 数 中 第 k 大 的 数字 。 

e 如 果 采 用 第 二 种 思路 ， 本 题 考 但 对 堆 、 红 黑 树 等 数据 结构 的 理 
解 。 当 需要 在 茶 数据 容 圳 内 频 迷 碍 找 及 蔡 换 最 大 值 时 ， 我 们 要 想到 二 又 
树 是 个 合适 的 选择 ， 并 能 想到 用 堆 或 者 红 黑 树 等 特殊 的 二 又 树 来 实现 。 

















面试 题 31: 连续 子 数 组 的 最 大 和 


题目 : 输入 一 个 整 型 数组 ， 数 组 里 有 正 数 也 有 负数 。 数 组 
中 一 个 或 连续 的 多 个 整数 组 成 一 个 子 数组 。 求 所 有 子 数组 的 和 
的 最 大 值 。 要 求 时 间 复 杂 度 为 O Cn) 。 

例如 输入 的 数组 为 {1,-2,3,10,-4,7,2,-5}， 和 最 大 的 子 数组 为 
{3,10,-4,7,2}， 因 此 输出 为 该 子 数组 的 和 18。 

看 到 这 道 题 ， 很 多 人 都 能 想到 最 直观 的 方法 ， 即 枚 举 出 数组 的 所 有 
子 数组 并 求 出 它们 的 和 。 一 个 长 度 为 n 的 数组 ， 总 共有 n (na 十 1) /2 个 子 
数组 。 计 算出 所 有 子 数组 的 和 ， 最 快 也 需要 O Cn) 的 时 间 。 通 名 最 直 
观 的 方法 不 会 是 最 优 的 解法 ， 面 试 官 将 提示 我 们 还 有 更 快 的 算法 。 


解法 一 : 举例 分 析 数 组 的 规律 


我 们 试 着 从 头 到 尾 逐 个 累加 示例 数组 中 的 每 个 数字 。 初 始 化 和 为 
0。 第 一 步 加 上 第 一 个 数字 1， 此 时 和 为 1。 接 下 来 第 二 步 加 上 数字 -2， 
和 就 变 成 了 -1。 第 三 步 加 上 数字 3。 我 们 注意 到 由 于 此 前 累计 的 和 是 -1， 
小 于 0， 那 如 果 用 -1 加 上 3， 得 到 的 和 是 2， 比 3 本 身 还 小 。 也 就 是 说 从 第 
一 个 数字 开始 的 子 数组 的 和 会 小 于 从 第 三 个 数字 开始 的 子 数组 的 和 。 
此 我 们 不 用 考虑 从 第 一 个 数字 开始 的 子 数 组 ， 之 前 累计 的 和 也 被 抛弃 。 

我 们 从 第 三 个 数字 重新 开始 累加 ， 此 时 得 到 的 和 是 3。 接 下 来 第 四 
步 加 10， 得 到 和 为 13。 第 五 步 加 上 -4， 和 为 9。 我 们 发 现 由 于 -4 是 一 个 负 
数 ， 因 此 累加 -4 之 后 得 到 的 和 比 原来 的 和 还 要 小 。 因 此 我 们 要 把 之 前 得 
到 的 和 13 保 存 下 来 ， 它 有 可 能 是 最 大 的 子 数组 的 和 。 第 六 步 加 上 数字 
7，9 加 7 的 结果 是 16， 此 时 和 比 之 前 最 大 的 和 13 还 要 大 ， 把 最 大 的 子 数 
组 的 和 由 13 更 新 为 16。 第 七 步 加 上 2， 累 加 得 到 的 和 为 18， 同 时 我 们 也 
要 更 新 最 大 子 数组 的 和 。 第 八 步 加 上 最 后 一 个 数字 -5， 由 于 得 到 的 和 为 
13， 小 于 此 前 最 大 的 和 18， 因 此 最 终 最 大 的 子 数 组 的 和 为 18， 对 应 的 子 
数组 是 {3,10,-4,7,2}。 整 个 过 程 可 以 用 表 5.2 总 结 如 下 : 

表 5.2 计算 数组 {1,-2,3,10,-4,7,2,-5} 中 子 数组 的 最 大 和 的 过 程 
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| 分 析 消 楚 之 后 ， 我 们 就 可 以 动手 写 代码 了 。 下 面 是 一 段 参考 


bool g_InvalidInput = false; 


int FindGreatestSumOfSubArray(int *pData, int nLength) 
{ 
£((pData == NULL) || (nLength <= 0) ) 
{ 
g_Invalidinput = true; 
return 0; 


} 


g_InvalidInput = false; 


int nCurSum = 0; 
int nGreatestSum = 0x80000000; 
for(int i = 0; i < nLength; ++i) 


{ 
if (nCurSum <= 0) 
nCurSum = pData[i]; 
else 
nCurSum += pData[i]; 


if(nCurSum > nGreatestSum) 
nGreatestSum = nCurSum; 


} 


return nGreatestSum; 
} 


面试 的 时 候 我 们 要 考虑 无 效 的 输入 ， 比 如 输入 的 数组 参数 为 空 指 
针 、 数 组 长 度 小 于 等 于 0 等 情况 。 此 时 我 们 让 函数 返回 什么 数字 ? 如 果 
是 返回 0， 那 我 们 又 怎么 区 分 子 数组 的 和 的 最 大 值 是 0 和 无 效 输 入 这 两 种 
不 同情 况 呢 ? 因此 我 们 定义 了 一 个 全 局 变量 来 标记 是 否 输 入 无 效 。 


解法 二 : 应 用 动态 规划 法 


如 果 算 法 的 功底 足够 扎实 ， 我 们 还 可 以 用 动态 规划 的 思想 来 分 析 这 
个 问题 。 如 果 用 函数 f G) 表示 以 第 i 个 数字 绪 尾 的 子 数 组 的 最 大 和 ， 那 
么 我 们 需要 求 出 max[f G) ]， 其 中 0<i<n。 我 们 可 用 如 下 递归 公式 求 
f G): 

















f= Di i= On (i-1) <0 
f (i-1)+ pDatali) i+ 0} Hf (i-1)>0 


这 个 公式 的 意义 : 当 以 第 i 一 1 个 数字 结尾 的 子 数 组 中 所 有 数字 的 和 
小 于 0 时 ， 如 果 把 这 个 负数 与 第 i 个 数 紧 加 ， 得 到 的 结果 比 第 i 个 数字 本 喘 
还 要 小 ， 所 以 这 种 情况 下 以 第 i 企 数字 结尾 的 子 数 组 融 是 第 i 个 数字 本 映 
(如 表 5.2 的 第 3 步 ) 。 如 果 以 第 i 一 1 个 数字 结尾 的 子 数 组 中 所 有 数字 的 
与 第 i 个 数字 累加 项 得 到 以 第 i 个 数字 结尾 的 子 数 组 中 所 有 数字 
JA o 
虽然 通 种 我 们 用 递归 的 方式 分 析 动 态 规划 的 问题 ， 但 最 终 都 会 基于 
循环 去 编码 。 上 述 公式 对 应 的 代码 和 前 面 给 出 的 代码 一 致 。 递 归公 式 中 
的 f G) 对 应 的 变量 是 nCurSum， 而 max[f (i) ] 就 是 nGreatestSum。 因 此 
可 以 说 这 两 种 思路 是 异曲同工 。 
源 代码 : 
本 题 完 整 的 源 代 码 详 见 31_GreatestSumOfSubarrays 项 目 。 
测试 用 例 : 


。 功能 测试 (输入 的 数组 中 有 正 数 也 有 负数 ， 输 入 的 数组 中 全 是 
正 数 ， 输 入 的 数组 中 全 是 负数 ) 。 

。 特殊 输入 测试 表示 数组 的 指针 为 NULL 指 针 ) 。 

本 题 考点 ; 

。 考查 对 时 间 复 杂 度 的 理解 。 这 道 题 如 果 应 聘 者 给 出 时 间 复 杂 度 
为 0 Gr) EEO (m) 的 算法 ， 是 不 能 通过 面试 的 。 

o 考查 对 动态 规划 的 理解 。 如 果 应 聘 者 熟练 掌握 了 动态 规划 算 
法 ， 那 么 他 就 能 轻松 地 找到 解 题 方案 。 如 果 没有 想到 用 动态 规划 的 思 
想 ， 那 么 应 聘 者 就 需要 仔细 地 分 析 累 加 子 数组 的 和 的 过 程 ， 从 而 找到 解 
题 的 规律 。 

。 考 杏 思维 的 全 面 性 。 能 否 合理 地 处 理 无 效 的 输入 ， 对 面试 结果 
有 很 重要 的 影响 。 


面试 题 32: 从 1 到 mn 整数 中 1 出 现 的 次 数 

题目 ， 输 入 一 个 整数 n， 求 从 1 到 n 这 n 个 整数 的 十 进 制 表示 
中 1 出 现 的 次 数 。 例 如 输入 12， 从 1 到 12 这 些 整 数 中 包含 1 的 数 
字 有 1，10，11 和 12，1 一 共 出 现 了 5 次 。 


不 考虑 时 间 效 率 的 解法 ， 靠 它 想 拿 Offer 有 点 难 

















如 果 在 面试 的 时 候 碰 到 这 个 问题 ， 应 聘 者 大 多 能 想到 最 直观 的 方 
法 ， 也 就 是 累加 1 到 n 中 每 个 整数 1 出 现 的 次 数 。 我 们 可 以 每 次 通过 对 10 
求 余 数 判断 整数 的 个 位 数字 是 不 是 1。 如 果 这 个 数字 大 于 10， 除 以 10 之 
后 再 判断 个 位 数字 是 不 是 1。 基 于 这 个 思路 ， 我 们 不 难 写 出 如 下 代码 : 
int NumberOflBetweenlAndN (unsigned int n) 


{ 


int number = 0; 
for (unsigned int i = 1; i <= n; ++ i) 
number += NumberOfl (i); 


return number; 


int NumberOfl(unsigned int n) 


int number = 0; 
while (n) 
if(n 3 10 == 1) 


number ++; 


n asn / 10; 


return number; 
} 

在 上 述 思 路 中 ， 我 们 对 每 个 数字 都 要 做 除法 和 求 余 运算 以 求 出 该 数 
字 中 1 出 现 的 次 数 。 如 果 输 入 数字 n，n 有 O Clogn) 位 ， 我 们 需要 判断 每 
一 位 是 不 是 1， 那 么 它 的 时 间 复 杂 度 是 O (Cnlogn) 。 当 输入 n 非 常 大 的 时 
修 ， 雷 要 大 量 的 计算 ， 运算 效率 不 高 。 面 试 官 不 会 满意 这 种 算法 ， 我 们 
仍然 需要 努力 。 


从 数字 规律 着 手 明 显 提 局 时 间 效 率 的 解法 ， 能 让 面试 官 耳目 
一 新 


如 果 和 希望 不 用 计算 每 个 数字 的 1 的 个 数 ， 那 就 只 能 去 寻找 1 在 数字 中 
出 现 的 规律 了 。 为 了 找到 规律 ， 我 们 不 妨 用 一 个 稍微 大 一 点 的 数字 比如 
21345 作 为 例子 来 分 析 。 我 们 把 从 1 到 21345 的 所 有 数字 分 为 两 段 ， 一 段 
是 从 1 到 1345， 男 一 段 是 从 1346 到 21345。 

我 们 先 看 从 1346 到 21345 中 1 出 现 的 次 数 。1 的 出 现 分 为 两 种 情况 。 

















首先 分 析 1 出 现在 最 高 位 〈 本 例 中 是 万 位 ) 的 情况 。 从 1346 到 21345 的 数 
字 中 ，1 出 现在 10000 一 19999 这 10000 个 数字 的 万 位 中 ， 一 共 出 现 了 
10000 (10) 个 。 

值得 注意 的 是 ， 并 不 是 对 所 有 5 位 数 而 言 在 万 位 出 现 的 次 数 都 是 
10000 个 。 对 于 万 位 是 1 的 数字 比如 输入 12345，1 只 出 现在 10000 一 12345 
的 万 位 ， 出 现 的 次 数 不 是 10 次 ， 而 是 2346 次 ， 也 就 是 除去 最 高 数字 之 后 
剩 下 的 数字 再 加 上 1 ( 即 2345 十 1=2346 次 ) 。 

接 下 来 分 析 1 出 现在 除 最 高 位 之 外 的 其 他 四 位 数 中 的 情况 。 例 子 中 
1346 一 21345 这 20000 个 数字 中 后 4 位 中 1 出 现 的 次 数 是 2000 次 。 由 于 最 高 
位 是 2， 我 们 可 以 再 把 1346 一 21345 分 成 两 段 ，1346 一 11345 和 11346 一 
21345。 每 一 段 剩 下 的 4 位 数字 中 ， 选 择 其 中 一 位 是 1， 其 余 三 位 可 以 在 0 
一 9 这 10 个 数字 中 任意 选择 ， 因 此 根据 排列 组 合 原则 ， 总 共 出 现 的 次 数 
是 2x10:=2000 次 。 

至 于 从 1 到 1345 中 1 出 现 的 次 数 ， 我 们 就 可 以 用 递归 求 得 了 。 这 也 是 
我 们 为 什么 要 把 1 一 21345 分 成 1 一 1345 和 1346 一 21345 两 段 的 原因 。 因 为 
把 21345 的 最 高 位 去 掉 就 变 成 1345， 便 于 我 们 采用 递归 的 思路 。 

基于 前 面 的 分 析 ， 我 们 可 以 写 出 如 下 代码 (为 了 编程 方便 ， 我 们 先 
把 数字 转换 成 字符 串 ) : 

















int NumberOflBetweenlAndN (int n) 
{ 


Eta <= 0) 
return 0; 


char strN[50]; 
sprintf(strN, "sd", n); 


return NumberOfl (strN); 
} 


int NumberOfl (const char* strN) 
{ 


TECISEPN | | ESIN E Ov I ESEN FQ | KEEN == eNO") 
return 0; 
int first SSErN = “es 


unsigned int length = static_cast<unsigned int>(strlen(strN) ); 


if (length == 1 && first == 0) 
return 0; 

L£ Glength=== T sé. first > 0) 
return i; 


// 假设 strn 是 "21345" 
// numFirstDigit £4k&F 10000-19999 的 第 一 个 位 中 的 数目 
int numFirstDigit = 0; 
Lf (first > 1) 
numFirstDigit = PowerBasel0 (length - 1); 
else if (first == 1) 
numFirstDigit = atoi(strN + 1) + 1; 


// numOtherDigits Æ 1346~21345 除了 第 一 位 之 外 的 数位 中 的 数目 

int numOtherDigits = first * (length-1) * PowerBasel0(length-2) ; 
// namRecursive £1~1345 中 的 数目 

int numRecursive = NumberOfl(strN + 1); 


return numFirstDigit + numOtherDigits + numRecursive; 


} 


int PowerBasel0 (unsigned int n) 
{ 
int result = 1; 
for (unsigned int i = 0; i < n; ++ i) 
result *= 10; 


return result; 








这 种 思路 是 每 次 去 挥 最 高 位 做 递归 ， 递 归 的 次 数 和 位 数 相同 。 一 个 
数字 n 有 O Cogn) 位 ， 因 此 这 种 思路 的 时 间 复 杂 度 是 O Clogn) ， 比 前 
面 的 原始 方法 要 好 很 多 。 





源 代码 : 
本 题 完 整 的 源 代 码 详 见 32_NumberOf1 项 目 。 
测试 用 例 : 


e 功能 测试 (输入 5、10、55、99 等 ) 。 
e 边界 值 测试 (输入 0、1 等 )。 
o 性 能 测试 (输入 较 大 的 数字 如 10000、21235 等 )。 


本 题 考点 : 


e 考查 应 聘 者 做 优化 的 激情 和 能 力 。 最 原始 的 方法 大 部 分 应 聘 者 
都 能 想到 。 当 面试 官 提示 还 有 更 快 的 方法 之 后 ， 应 聘 者 千 万 不 要 轻易 放 
弃 答 试 。 虽 然 想 出 O Cogn) 的 方法 不 容易 ， 但 应 聘 者 要 展示 自己 退 求 
更 快 算法 的 激情 ， 多 尝试 不 同 的 方法 ， 必 要 的 时 候 可 以 要 求 面试 官 给 出 
提示 ， 但 不 能 轻易 说 自己 想 不 出 来 并 且 放 弃 努 力 。 

e 考查 面 应 聘 者 对 复杂 问题 的 思维 能 力 。 要 想 找到 O Cogn) 的 方 
法 ， 应 聘 者 需要 有 很 严密 的 数学 思维 能 力 ， 并 且 还 要 通过 分 析 具 体例 子 
e a a 

常 有 用 。 


面试 题 33: 把 数组 排 成 最 小 的 数 


题目 : 输入 一 个 正 整数 数组 ， 把 数组 里 所 有 数字 拼接 起 来 
排 成 一 个 数 ， 打 印 能 拼接 出 的 所 有 数字 中 最 小 的 一 个 。 例 如 输 
入 数组 {3,32,321}， 则 打印 出 这 3 个 数字 能 排 成 的 最 小 数字 
321323. 

这 个 题目 最 直接 的 做 法 应 该 是 先 求 出 这 个 数组 中 所 有 数字 的 全 排 
列 ， 然 后 把 每 个 排列 拼 起 来 ， 最 后 求 出 拼 起 来 的 数字 的 最 大 值 。 求 数组 
的 排列 和 面试 题 28“ 字 符 串 的 排列 ”非常 类 似 ， 这 里 不 再 详细 介绍 。 根 据 
排列 组 合 的 知识 ，n 个 数字 总 共有 n! 个 排列 。 我 们 再 来 看 一 种 更 快 的 算 


Ea 
这 道 题 其 实 是 布 望 我 们 能 找到 一 个 排 友 规则， 数组 根据 这 个 规则 排 























序 之 后 能 排 成 一 个 最 小 的 数字 。 要 确定 排序 规则 ， 就 要 比较 两 个 数字 ， 
也 束 是 给 出 两 个 数字 m 和 n， 我 们 需要 确定 一 个 规则 判断 m 和 n 哪 个 应 该 
排 在 前 面 ， 而 不 是 仅仅 比较 这 两 个 数字 的 值 哪个 更 大 。 

根据 题目 的 要 求 ， 两 个 数字 m 和 n 能 拼接 成 数字 mn 和 nm。 如 果 
mn<nm， 那 么 我 们 应 该 打印 出 mn， 也 就 是 m 应 该 排 在 n 的 前 面 ， 我 们 定 
义 此 时 m 小 于 n; 反之 ， 如 果 nm<mn， 我 们 定义 hn 小 于 mm。 如 果 mn=nm， 
m 等 于 n。 在 下 文中 ， 符 号 <<”、“>” 及 “=” 表 示 常 规 意义 的 数值 的 大 小 关 
系 ， 而 文字 “大 于 ”“ 小 于 ” “等 于 ”表示 我 们 新 定义 的 大 小 关系 。 

接 下 来 考虑 怎么 去 拼接 数字 ， 即 给 出 数字 m 和 mn， 怎么 得 到 数字 mn 
和 nm 并 比较 它们 的 大 小 。 直 接 用 数值 去 计算 不 难 办 到 ， 但 需要 考虑 到 
一 个 潜在 的 问题 就 是 nm 和 n 都 在 int 能 表达 的 范围 内 ， 但 把 它们 拼 起 来 的 
数字 mn 和 nm 用 int 表 示 就 有 可 能 溢出 了 ， 所 以 这 还 是 一 个 隐形 的 大 数 问 


一 个 非常 直观 的 解决 大 数 问题 的 方法 就 是 把 数字 转换 成 字符 串 。 忆 
外 ， 由 于 把 数字 m 和 n 拼 接 起 来 得 到 mn 和 nm， 它 们 的 位 数 上 朋 定 是 相同 
的 ， 因 此 比较 它们 的 大 小 只 需要 按照 字符 串 大 小 的 比较 规则 就 可 以 了 。 

基于 这 个 思路 ， 我 们 可 以 写 出 如 下 代码 : 














const int g MaxNumberLength = 10; 


char* g StrCombinel 


$y new char[g MaxNumberLength * 
char* g StrCombine2 


new char[g MaxNumberLength * 


bo bo 


void PrintMinNumber(int* numbers, int length) 


ee 
el |e 


strNumber2) 


{ 
if (numbers == NULL || length <= 0) 
return; 
char** strNumbers = (char**) (new int[length]); 
for(int i = 0; i < length; ++1) 
{ 
strNumbers[i] = new char[g MaxNumberLength + 1]; 
sprintf (strNumbers[i], "sd", numbers[i]); 
} 
qsort(strNumbers, length, sizeof(char*), compare); 
for(int i = 0; i < length; ++i) 
printf("%s", strNumbers[i]); 
PLInNtE ("in"): 
for(int i= 0; i < length; ++1) 
delete[] strNumbers [il]; 
delete[] strNumbers; 
} 
int compare(const void* strNumberl, const void* 
{ 
strcpy(g StrCombinel, * (const char**)strNumber1) ; 
strcat(g StrCombinel, *(const char**)strNumber2) ; 
strcpy(g StrCombine2, *(const char**) strNumber2) ; 
strcat(g StrCombine2, *(const char**)strNumber1); 


y 
i 


return strcemp(g StrCombinel, g StrCombine2) ; 


二 


} 


在 上 述 代 码 中 ， 我 们 先 把 数组 中 的 整数 转换 成 字符 串 ， 在 函数 


compare 中 定义 比较 规则 ， 并 根据 该 规则 用 库 函 数 qsort 排 序 。 最 后 把 排 


好 序 的 数组 中 的 数字 依次 打印 出 来 ， 束 是 该 数组 中 数 


Poe 


Fe 


拼接 出 来 的 最 


小 数字 。 这 种 思路 的 时 间 复 杂 度 和 qsort 的 时 间 复 杂 度 相同 ， 也 就 是 
O (nlogn) ， 这 比 用 n! 的 时 间 求 出 所 有 排列 的 思路 要 好 很 多 。 
上 述 思 路 中 ， 我 们 定义 了 一 种 新 的 比较 两 个 数 的 规则 ， 这 种 规则 是 


不 是 有 效 的 ? 另外 ， 我 们 只 是 定义 了 比较 两 个 数 的 规则 ， 却 用 它 来 排序 
一 个 含有 多 个 数字 的 数组 ， 最 终 拼接 数组 中 的 所 有 数字 得 到 的 是 不 是 真 
的 就 是 最 小 的 数字 ? 一 些 严 格 的 面试 官 还 会 要 求 我 们 给 出 严格 的 数学 证 
明 ， 以 确保 我 们 的 解决 方案 是 正确 的 。 

我 们 首先 证 明之 前 定义 的 比较 两 个 数字 大 小 的 规则 是 有 效 的 。 一 个 
i 2 自 反 性 、 对 称 性 和 传递 性 。 我 们 分 别 予 
以 证 明 。 

(1) 自 反 性 : 显然 有 aa=aa， 所 以 a 等 于 a。 

(2) 对 称 性 : 如 果 a 小 于 b， 则 ab<ba， 所 以 ba>ab， 因 此 b 大 于 a。 

(3) 传递 性 如果 a 小 于 b， 则 ab<ba。 假 设 a 和 b 用 十 进 制 表示 时 分 
别 有 ] 位 和 m 位 ， 于 是 ab=ax10" 十 b，ba=bx10' 十 a。 

ab<ba > ax10"+b<bx10'+a — ax10"—a< bx10'—b 

+a (10"-1) <b (10 一 1) Sa/ (10 一 1) <b/ (10"—1) 

同时 如 果 b 小 于 c， 则 bc<cb。 假 设 c 用 十 进 制 表示 是 有 n 位 ， 和 前 面 
的 证 明 过 程 一 样 ， 可 以 得 到 b/ (10" 一 1) <c/ 〈10" 一 1) 。 

a/ (10'—1) <b/ (10" 一 1) 并 有 Hb/ (10"—-1) <c/ (10' 一 1) =a/ (10! 
一 1) <c/ (1œ—1) Sa (10" 一 1) <c (10'—1) 

—ax10"+c<cx10' +a > ac<ca >» a/ħ Fc 

于 是 我 们 证 明了 这 种 比较 规则 满足 自 反 性 、 对 称 性 和 传递 性 ， 是 一 
种 有 效 的 比较 规则 。 接 下 来 我 们 证 明 根 据 这 种 比较 规则 把 数组 排序 之 
后 ， 把 数组 中 的 所 有 数字 拼接 起 来 得 到 的 数字 的 确 是 最 小 的 。 直 接 证 明 
不 是 很 容易 ， 我 们 不 妨 用 反 证 法 来 证 明 。 

我 们 把 n 个 数 按照 前 面 的 排序 规则 排序 之 后 ， 表 示 为 AI1A>A3...A。 
假设 这 样 拼接 出 来 的 数字 并 不 是 最 小 的 ， 即 至 少 存 在 两 个 x 和 
y (O<x<y<n) ， 交 换 第 x 个 数 和 第 y 个 数 后 ，AlAs...Ay.…Ax.… 
An<A1A2...Ax...Ay...Ane 

由 于 AiA2.…Ax…Ay.…An 是 按照 前 面 的 规则 排 好 的 序列 ， 所 以 有 A、 
TA A A A A TA 

由 于 Ay- 小 于 Ay， 所 以 Ay_1Ay<AyAy-1。 我 们 在 序列 A1A2..….Ax.… 
Ay_1Ay.…An 中 交换 A,_1 和 A,， 有 Ai1A2.….Ax…Ay_1Ay.…An<A1A2..… 
Ax…AyAy-1.…An〔 这 个 实际 上 也 需要 证 明 ， 感 兴趣 的 读者 可 以 自己 试 
着 证 明 ) 。 我 们 就 这 样 一 直 把 A, 和 前 面 的 数字 交换 ， 直 到 和 Ax 交 换 为 
tke PE REA A,Ad...Ay...Ay—Ay..-Ap<AyAg..-Ay.-AyAy—y.. 
A,<A,Ag...AX...AyAy—Ay—1.--AnS...<AyAy.. -AyAy..-Ay—2Ay—1.--Ane 

FIFE PAZ) FA, +1, PUAA 1 1<Ay4 Aye RIEF IAA... 














AJAxAx+1Ay-2Ay-1An 中 只 交换 Ax 和 Ax+1， 有 AIA2...AyAxAx+1… 
Ay-2Ay-TAn<AIA2AyAxiiAxc…Ay-2Ay-TeAn。 RZ PRE 
拿 A、 和 它 后 面 的 数字 交换 ， 直 到 和 Ay_1 交 换 为 止 。 于 是 就 有 A1A2.… 
AVAAxH1Ay 2Ay 1.An<AIA2.AVAxTIAxAy 2Ay AnS. 
<A, Ap... AyA,4+1A,49---Ay—2Ay— Ay. Ane 

所 以 AIA>.…AxAy An<AIA?.…Ay.Ax….An， 这 和 我 们 的 假设 的 
Ay Ap...Ay..Ay..AgsAyAy...Ay.. Ay. .An 相 矛 盾 。 

所 以 假设 不 成 立 ， 我 们 的 算法 是 正确 的 。 

源 代码 : 

本 题 完 整 的 源 代码 详 见 33_SortArrayForMinNumber 项 目 。 

测试 用 例 : 

o 功能 测试 《输入 的 数组 中 有 多 个 数字 ， 输 入 的 数组 中 的 数字 有 
重复 的 数位 ， 输 入 的 数字 只 有 一 个 数字 ) 。 

e 特殊 输入 测试 (表示 数组 的 指针 为 NULL 指 针 〉。 


本 题 考点 : 


e 本 题 有 两 个 难点 ;第 一 个 难点 是 想 出 一 种 新 的 比较 规则 来 排序 
一 个 数组 ;第 二 个 难点 在 于 证 明 这 个 比较 规则 是 有 效 的 ， 并 且 证 明 根 据 
这 个 规则 排序 之 后 把 数组 中 所 有 数字 拼接 起 来 得 到 的 数字 是 最 小 的 。 要 
想 解决 这 两 个 难点 ， 都 要 求 应 聘 者 有 很 强 的 数学 功 捕 和 逻辑 思维 能 力 。 

。 考查 解决 大 数 问 题 的 能 力 。 应 聘 者 在 面试 的 时 候 要 意识 到 ， 把 
两 个 int 型 的 整数 拼接 起 来 得 到 的 数字 可 能 会 超出 int 型 数字 能 够 表达 的 范 
围 ， 从 而 导致 数字 洲 出 。 我 们 可 以 用 字符 串 表 示 数 字 ， 这 样 束 能 简洁 地 
解决 大 数 问题 。 


5.3 ”时 间 效 率 与 空间 效率 的 平衡 


人 硬件 的 及 展 一 直 遵 循 看 摩尔 定律 ， 内 存 的 容量 基本 上 每 阳 18 个 月 残 
会 翻 一 和 看。 由 于 内 存 的 容量 增加 迅速 ， 在 软件 开发 的 过 程 中 我 们 允许 以 
牺牲 一 定 的 空间 为 代码 来 优化 时 间 性 能 ， 以 尽 可 能 地 缩短 软件 的 啊 应 时 
间 。 这 就 是 我 们 通常 所 说 的 “以 空间 换 时 间 ?”。 

在 面试 的 时 候 ， 如 采 我 们 分 配 少量 的 辅助 空间 来 保存 计算 的 中 间 结 
果 以 提高 时 间 效 率 ， 这 样 的 思路 通常 是 可 以 接受 的 。 本 书 中 收集 的 面试 























题 中 有 不 少 这 种 类 型 的 题目 ， 比 如 在 面试 题 34“ 丑 数 ” 中 用 一 个 数组 按照 
从 小 到 大 的 顺序 保存 已 经 求 出 的 丑 数 ;在 面试 题 43 人 个 仍 子 的 点 数 ” 中 
交 蔡 使 用 两 个 数组 求 般 子 每 个 点 数 出 现 的 次 数 。 

值得 注意 的 是 , “以 时 间 换 空间 ”的 策略 并 不 一 定 都 是 可 行 的 ， 在 面 
试 的 时 候 要 具体 问题 具体 分 析 。 我 们 都 知道 在 n 个 无 序 的 元 素 里 做 查找 
操作 ， 需 要 O Cn) 的 时 间 。 但 如 果 我 们 把 这 些 元 素 放 进 一 个 哈 希 表 ， 
那么 在 哈 希 表 内 就 能 实现 O (1) 的 查找 。 但 同时 实现 一 个 哈 希 表 是 有 
空间 消耗 的 ， 是 不 是 值得 以 多 消耗 空间 为 前 提 来 换取 时 间 性 能 的 提升 ， 
我 们 需要 根据 实际 情况 仔细 权衡 。 在 面试 题 35“ 第 一 个 只 出 现 一 次 的 字 
符 ” 中 ， 我 们 用 数组 实现 了 一 个 简易 哈 希 表 ， 有 了 这 个 哈 希 表 就 能 实现 
O (1) 查找 任意 字符 。 对 于 ASCII 码 的 字符 而 言 ， 总 共 只 有 256 个 字 
符 ， 因 此 只 需要 1K 的 辅助 内 存 。 这 点 内 存 消 耗 对 于 绝 大 多 数 硬 件 来 说 
是 完全 可 以 接受 的 。 但 如 果 是 16 位 的 Unicode 的 字符 ， 创 建 这 样 一 个 长 
度 为 2* 的 整 型 数组 需要 4x2* 也 就 是 256K 的 内 存 。 这 对 于 个 人 电脑 来 说 也 
是 可 以 接受 的 ， 但 对 于 一 些 嵌 入 式 的 开发 就 要 慎重 了 。 

很 多 时 候 时 间 效 率 和 空间 效率 存在 类 似 于 鱼 与 能 掌 的 关系 ， 我 们 需 
要 在 它们 之 间 有 所 取舍 。 在 面试 的 时 候 究竟 是 “以 时 间 换 空间 ”还 是 “以 
空间 换 时 间 ”， 我 们 可 以 和 面试 官 进 行 探 讨 。 多 和 面试 官 进行 这 方面 的 
讨论 是 很 有 必要 的 ， 这 既 能 显示 我 们 的 沟通 能 力 ， 又 能 展示 我 们 对 软件 
性 能 全 方位 的 把 握 能 力 。 


面试 题 34: HAY 


题目 : 我 们 把 只 包含 因子 2、3 和 5 的 数 称 作 丑 数 (Ugly 
Number) 。 求 按 从 小 到 大 的 顺序 的 第 1500 个 丑 数 。 例 如 6、8 
都 是 丑 数 ， 但 14 不 是 ， 因 为 它 包含 因子 7。 习 惯 上 我 们 把 1 当做 
第 一 个 丑 数 。 


逐个 判断 每 个 整数 是 不 是 丑 数 的 解法 ， 直 观 但 不 够 高 效 


所 谓 一 个 数 mm 是 另 一 个 数 n 的 因 了 于 ， 征 指 n 能 被 m 整 除 ， 也 束 是 n%m 
二 0。 根 据 丑 数 的 定义 ， 丑 数 只 能 被 >、3 和 5 整除 。 也 就 是 说 如 果 一 个 数 
能 被 2 整除 ， 我 们 把 它 连 续 除 以 2， 如 果 能 被 3 整除 ， 束 连续 除 以 3;， 如 果 
能 被 5 整除 ， 就 除 以 连续 5。 如 果 最 后 我 们 得 到 的 是 1， 那 么 这 个 数 就 是 
AR, AMA. 

因此 我 们 可 以 写 出 下 面 的 函数 来 判断 一 个 数 是 不 是 丑 数 : 















































bool IsUgly(int number) 


{ 
while (number % 2 == 0) 
number /= 2; 
while(number % 3 == 0) 
number /= 3; 
while(number % 5 == 0) 


number /= 5; 


return (number == 1) ? true : false; 


接 下 来 ， 我 们 只 需要 按照 顺序 判断 每 一 个 整数 是 不 是 丑 数 ， 即 : 


int GetUglyNumber (int index) 





{ 
if (index <= 0) 
return 0; 
int number = 0; 
int uglyFound = 0; 
while (uglyFound < index) 
{ 
++number; 
if (IsUgly (number) ) 
{ 
++uglyFound; 
} 
} 
J 
return number; 
} 


我 们 只 需要 在 函数 GetUglyNumber 中 传 入 参数 1500， 就 能 得 到 第 
1500 个 丑 数 。 该 算法 非常 直观 ， 代 码 也 非常 简洁 ， 但 最 大 的 问题 每 个 整 
数 都 需要 计算 。 即 使 一 个 数字 不 是 丑 数 ， 我 们 还 是 需要 对 它 做 求 余 数 和 
除法 操作 。 因 此 该 算法 的 时 间 效 率 不 是 很 高 ， 面 试 官 也 不 会 就 此 满足 ， 
他 会 提示 我 们 还 有 更 高 效 的 算法 。 


创建 数组 保存 已 经 找到 的 丑 数 ， 用 空间 换 时 间 的 解法 


前 面 的 算法 之 所 以 效率 低 ， 很 大 程度 上 是 因为 不 管 一 个 数 是 不 是 丑 
数 我 们 对 它 都 要 作 计 算 。 接 下 来 我 们 试 着 找到 一 种 只 要 计算 丑 数 的 方 

















法 ， 而 不 在 非 丑 数 的 整数 上 花费 时 间 。 根 据 丑 数 的 定义 ， 丑 数 应 该 是 男 
一 个 丑 数 乘 以 2、3 或 者 5 的 结果 〈1 除 外 ) 。 因 此 我 们 可 以 创建 一 个 数 
组 ， 里 面 的 数字 是 排 好 序 的 丑 数 ， 每 一 个 丑 数 都 是 前 面 的 丑 数 乘 以 2、3 
或 者 5 得 到 的 。 

这 种 思路 的 关键 在 于 怎样 确保 数组 里 面 的 丑 数 是 排 好 序 的 。 假 设 数 
组 中 已 经 有 知 干 个 丑 数 排 好 序 后 存放 在 数组 中 ， 并 且 把 已 有 最 大 的 丑 数 
记 做 M， 我 们 接 下 来 分 析 如 何 生成 下 一 个 丑 数 。 该 丑 数 肯定 是 前 面 某 一 
个 丑 数 乘 以 2、3 或 者 5 的 结果 ， 所 以 我 们 首先 考虑 把 已 有 的 每 个 丑 数 乘 
以 2。 在 乘 以 2 的 时 候 ， 能 得 到 若干 个 小 于 或 等 于 M 的 结果 。 由 于 是 按照 
顺序 生成 的 ， 小 于 或 者 等 于 M 肯 定 已 经 在 数组 中 了 ， 我 们 不 需 再 次 考 
les 还 会 得 到 若干 个 大 于 M 的 结果 ， 但 我 们 只 需要 第 一 个 大 于 M 的 结 
果 ， 因 为 我 们 希望 丑 数 是 按 从 小 到 大 的 顺序 生成 的 ， 其 他 更 大 的 结果 以 
后 再 说 。 我 们 把 得 到 的 第 一 个 乘 以 2 后 大 于 M 的 结果 记 为 M*。 同 样 ， 我 
们 把 已 有 的 每 一 个 丑 数 乘 以 3 和 5， 能 得 到 第 一 个 大 于 M 的 结果 M3 和 
M。。 那 么 下 一 个 丑 数 应 该 是 M,、M3a 和 M。 这 3 个 数 的 最 小 者 。 

前 面 分 析 的 时 候 ， 提 到 把 已 有 的 每 个 丑 数 分 别 都 乘 以 2、3 和 5。 事 
实 上 这 不 是 必须 的 ， 因 为 已 有 的 丑 数 是 按 顺 序 存放 在 数组 中 的 。 对 乘 以 
2 而 言 ， 肯 定 存 在 某 一 个 丑 数 T,， 排 在 它 之 前 的 每 一 个 丑 数 乘 以 2 得 到 的 
结果 都 会 小 于 已 有 最 大 的 丑 数 ， 在 它 之 后 的 每 一 个 丑 数 乘 以 2 得 到 的 结 
果 都 会 太 大 。 我 们 只 需 记 下 这 个 丑 数 的 位 置 ， 同 时 每 次 生成 新 的 丑 数 的 
时 候 ， 去 更 新 这 个 T,。 对 乘 以 3 和 5 而 言 ， 也 存在 着 同样 的 T; 和 T。。 

有 了 这 些 分析 ， 我 们 就 可 以 写 出 如 下 代码 : 





























int GetUglyNumber Solution2(int index) 


{ 


} 


int Min(int numberl, int number2, 


{ 


} 


算 ， 因 此 时 间 效 率 有 明显 提升 。 但 也 需要 指出 ， 


if (index <= 0) 


return 0; 
int *pUglyNumbers = new int[index]; 
pUglyNumbers[0] = 1; 
int nextUglyIndex = 1; 
int *pMultiply2 = pUglyNumbers; 
int *pMultiply3 = pUglyNumbers; 
int *pMultiply5 = pUglyNumbers; 


while (nextUglyIndex < index) 


{ 


} 


int 


int min = Min(*pMultiply2 * 2, 
otal ymabera mextogi ymist] = 


while(*pMultiply2 * 2 
++pMultiply2; 
while(*pMultiply3 * 

++pMultiply3; 
while (*pMultiply5 
++pMultiply5; 


++nextUglyIndex; 


ugly 


delete[] pUglyNumbers; 
return ugly; 


int 


min 


min = 
= (min < number3) 


return min; 


和 第 


一 种 思路 相 比 ， 


(numberl < number2) 


2 min: 


min; 


*pMultiply3 * 3, 


*pMultiply5 * 5); 


pUglyNumbers [nextUglyIndex] ) 


pUglyNumbers [nextUglyIndex] ) 


? numberl 
number3; 





= pUglyNumbers [nextUglyIndex] ) 


= pUglyNumbers [nextUglyIndex - 1]; 


int number3) 


: number2; 


第 二 种 思路 不 需要 在 非 丑 数 的 整数 上 做 任何 计 


第 二 种 算法 由 于 需要 保 


存 已 经 生成 的 丑 数 ， 因 此 需要 一 个 数组 ， 从 而 增加 了 空间 消耗 。 如 果 是 
求 第 1500 个 丑 数 ， 将 创建 一 个 能 容纳 1500 个 丑 数 的 数组 ， 这 个 数组 占 内 
存 6KB。 而 第 一 种 思路 没有 这 样 的 内 存 开销 。 忌 的 来 说 ， 第 二 种 思路 相 
当 于 用 较 小 的 空间 消耗 换取 了 时 间 效 率 的 提升 。 


源 代码 : 
本 题 完整 的 源 代码 详 见 34_UglyNumber 项 目 。 
测试 用 例 : 


e 功能 测试 (输入 2、3、4、5、6 等 ) 。 
e 特殊 输入 测试 (边界 值 1、 无 效 输入 0) 。 
e 性 能 测试 〈 输 入 较 大 的 数字 ， 如 1500) 。 


本 题 考点 : 


o 考研 应 聘 者 对 时 间 复 杂 度 的 理解 。 绝 大 部 分 应 聘 者 都 能 想 出 第 
一 种 思路 。 在 面试 官 提示 还 有 更 快 的 解法 之 后 ， 应 聘 者 能 否 分 析出 时 间 
效率 的 瓶 令 ， 并 找 出 解决 方案 ， 是 能 人 否 通过 这 轮 面试 的 关键 。 

e 考查 应 聘 者 的 学 习 能 力 和 沟通 能 力 。 丑 数 对 很 多 人 而 言 古 个 新 
概念 。 有 些 面 试 官 喜欢 在 面试 的 时 候 定 义 一 个 新 概念 ， 然 后 针对 这 个 新 
概念 出 面试 题 。 这 束 要 求 应 聘 者 听 到 不 熟悉 的 概念 之 后 ， 要 有 主动 积极 
的 态度 ， 大 胆 同 面试 官 提 问 ， 经 过 几 次 思考 、 提 问 、 再 思考 的 循环 ， 在 
短 时 间 内 理解 这 个 新 概念 。 这 个 过 程 就 体现 了 应 聘 者 的 学 习 能 力 和 沟通 


能 力 。 
面试 题 35: 第 一 个 只 出 现 一 次 的 字符 


题目 : 在 字符 串 中 找 出 第 一 个 只 出 现 一 次 的 字符 。 如 输 
入 "abaccdeff"， 则 输出 'b'。 

看 到 这 道 题 时 ， 我 们 最 直观 的 想法 是 从 头 开始 扫描 这 个 字符 串 中 的 
每 个 字符 。 当 访问 到 某 字 符 时 拿 这 个 字符 和 后 面 的 每 个 字符 相 比 较 ， 如 
果 在 后 面 没 有 发 现 重 复 的 字符 ， 则 该 字符 就 是 只 出 现 一 次 的 字符 。 如 果 
字符 串 有 n 个 字符 ， 每 个 字符 可 能 与 后 面 的 O (Cn) 个 字符 相 比 较 ， 因 此 
这 种 思路 的 时 间 复 杂 度 是 O m) 。 面 试 官 不 会 满意 这 种 思路 ， 他 会 提 
示 我 们 还 有 更 快 的 方法 。 

由 于 题目 与 字符 出 现 的 次 数 相关 ， 我 们 是 不 是 可 以 统计 每 个 字符 在 
该 字符 串 中 出 现 的 次 数 ? 要 达到 这 个 目的 ， 我 们 需要 一 个 数据 容器 来 存 









































放 每 个 字符 的 出 现 次 数 。 在 这 个 数据 容器 中 可 以 根据 字符 来 查找 它 出 现 
的 次 数 ， 也 就 是 说 这 个 容器 的 作用 是 把 一 个 字符 映射 成 一 个 数字 。 在 常 
用 的 数据 容器 中 ， 哈 希 表 正 是 这 个 用 途 。 

为 了 解决 这 个 问题 ， 我 们 可 以 定义 哈 希 表 的 键 值 (Key) 是 字符 ， 
而 值 (Value) 是 该 字符 出 现 的 次 数 。 同 时 我 们 还 需要 从 头 开 始 扫 描 字 
符 串 两 次 。 第 一 次 扫描 字符 串 时 ， 每 扫描 到 一 个 字符 就 在 哈 希 表 的 对 应 
项 中 把 次 数 加 1。 接 下 来 第 二 次 扫描 时 ， 每 扫描 到 一 个 字符 就 能 从 哈 希 
和 
《 All i 

哈 希 表 是 一 种 比较 复杂 的 数据 结构 ， 并 且 C++ 的 标准 模板 库 中 没有 
实现 哈 希 表 。 接 下 来 我 们 要 考虑 的 问题 就 是 如 何 实现 哈 希 表 。 由 于 本 题 
的 特殊 性 ， 我 们 只 需要 一 个 非常 简单 的 哈 希 表 就 能 满足 要 求 。 字 符 
(char) 是 一 个 长 度 为 8 的 数据 类 型 ， 因 此 总 共有 256 种 可 能 。 于 是 我 们 
创建 一 个 长 度 为 256 的 数组 ， 每 个 字母 根据 其 ASCII 码 值 作 为 数组 的 下 标 
对 应 数组 的 一 个 数字 ， 而 数组 中 存储 的 是 每 个 字符 出 现 的 次 数 。 这 样 我 
们 就 创建 了 一 个 大 小 为 256， 以 字符 ASCII 码 为 键 值 的 哈 希 表 。 

第 一 次 扫描 时 ， 在 哈 希 表 中 更 新 一 个 字符 出 现 的 次 数 的 时 间 是 
O (1) 。 如 果 字 符 串 长 度 为 n， 那 么 第 一 次 扫描 的 时 间 复 杂 度 是 
O Cn) 。 第 二 次 扫描 时 ， 同 样 O (1) 能 读 出 一 个 字符 出 现 的 次 数 ， 所 
以 时 间 复 杂 度 仍然 是 O (Cn) 。 这 样 算 起 来 ， 总 的 时 间 复 杂 度 是 
O Cn) 。 同 时 ， 我 们 需要 一 个 包含 256 个 字符 的 辅助 数组 ， 它 的 大 小 是 
1K。 由 于 这 个 数组 的 大 小 是 个 常数 ， 因 此 可 以 认为 这 种 算法 的 空间 复 
杂 度 是 O (1) 。 

当 我 们 向 面试 官 讲述 清楚 这 个 思路 并 得 到 面试 官 的 首肯 之 后 ， 就 可 
以 动手 写 代码 了 。 下 面 是 一 段 参考 代码 : 

















char FirstNotRepeatingChar(char* pString) 


{ 


if(pString == NULL) 
return '\0O'; 
const int tableSize = 256; 
unsigned int hashTable[tableSize]; 
for(unsigned int i = 0; i<tableSize; ++ i) 
hashTable[i] = 0; 


char* pHashKey = pString; 
while (* (pHashKey) != '\0") 
hashTable[* (pHashKey++)] ++; 


pHashKey = pString; 
while (*pHashKey != '\0"') 
{ 
if (hashTable[*pHashKey] == 1) 
return *pHashKey; 


pHashKey++; 
} 


return '\0'; 


源 代码 : 
本 题 完整 的 源 代 人 码 详 见 35_FirstNotRepeatingChar 项 目 。 
测试 用 例 : 


o 功能 测试 “字符 串 中 存在 只 出 现 一 次 的 字符 ， 字 符 串 中 不 存在 
只 出 现 一 次 字符 ， 字 符 串 中 所 有 字符 都 只 出 现 一 次 ) 。 
。 特殊 输入 测试 (字符 串 为 NULL 指 针 〉。 


本 题 考点 : 


o 考查 对 数组 、 字 符 串 的 编程 能 力 。 

e 考查 对 哈 希 表 的 理解 及 运用 。 

o 考查 对 时 间 效 率 及 空间 效率 的 分 析 能 力 。 当 面试 官 提示 最 直观 
的 算法 不 是 最 优 解 的 时 候 ， 应 聘 者 需要 立即 分 析出 这 种 算法 的 时 间 效 
率 。 在 想 出 基于 哈 希 表 的 算法 之 后 ， 应 聘 者 也 应 该 分 析出 该 方法 的 时 间 
效率 和 空间 效率 分 别 是 O Cn) 和 O (1) 。 

















本 题 扩展 : 


在 前 面 的 例子 中 ， 我 们 之 所 以 可 以 把 哈 希 表 的 大 小 设 为 256， 是 因 
为 字符 (char) 是 8 个 bit 的 类 型 ， 总 共 只 有 256 个 字符 。 但 实际 上 字符 不 
只 是 256 个 ， 比 如 中 文 就 有 几 千 个 汉字 。 如 果 题目 要 求 考虑 汉字 ， 前 面 
的 算法 是 不 是 有 问题 ? 如 果 有 ， 可 以 怎么 解决 ? 











相关 题目 ; 
。 定义 一 个 函数 ， 输 入 两 个 字符 串 ， 从 第 一 个 字符 串 中 删除 在 第 
二 个 字符 串 中 出 现 过 的 所 有 字符 。 例 如 从 第 一 个 字符 串 "We are 


students." 中 删除 在 第 二 个 字符 串 "aeiou" 中 出 现 过 的 字符 得 到 的 结果 
是 Wr Stdnts. "。 为 了 解决 这 个 问题 ， 我 们 可 以 创建 一 个 用 数组 实现 的 
简单 哈 希 表 来 存储 第 二 个 字符 串 。 这 样 我 们 从 头 到 尾 扫描 第 一 个 字符 串 
的 每 一 个 字符 时 ， 用 O (1) 时 间 束 能 判断 出 该 字符 是 不 是 在 第 二 个 字 
符 中 。 如 果 第 一 个 字符 串 的 长 度 是 n， 那 么 总 的 时 间 复 杂 度 是 DO Cn) 。 

e 定义 一 个 函数 ， 删 除 字 符 串 中 所 有 重复 出 现 的 字符 。 例 如 输 
入 "google"， 删 除 重复 的 字符 之 后 的 结果 是 "gole"。 这 个 题目 和 上 面 的 问 
题 比 较 类 似 ， 我 们 可 以 创建 一 个 用 布尔 型 数组 实现 的 简单 的 哈 希 表 。 数 
组 中 的 元 素 的 意义 是 其 下 标 看 做 ASCII 码 后 对 应 的 字母 在 字符 串 中 是 否 
己 经 出 现 。 我 们 先 把 数组 中 所 有 的 元 素 都 设 为 false。 以 "google" 为 例 ， 
当 扫 描 到 第 一 个 g 时 ，g 的 ASCII 码 是 103， 那 么 我 们 把 数组 中 下 标 为 103 
的 元 素 设 为 tue。 当 扫 摘 到 第 二 个 g 时 ， 我 们 发 现 数组 中 下 标 为 103 的 元 
素 的 值 是 true， 就 知道 g 在 前 面 已 经 出 现 了 。 也 就 是 说 ， 我 们 用 O (1) 
时 间 就 能 判断 出 每 个 字符 是 否 在 前 面 已 经 出 现 过 。 如 果 字 符 串 的 长 度 是 
n， 那 么 总 的 时 间 复 杂 度 是 O Cn) 。 

e 在 英语 中 ， 如 果 两 个 单词 中 出 现 的 字母 相同 ， 并 且 每 个 字母 出 
现 的 次 数 也 相同 ， 那 么 这 两 个 单词 互 为 变 位 词 (Anagram) 。 例 如 silent 
与 listen、evil 与 live 等 互 为 变 位 词 。 请 完成 一 个 函数 ， 判 断 输 入 的 两 个 字 
符 串 是 不 是 互 为 变 位 词 。 我 们 可 以 创建 一 个 用 数组 实现 的 简单 哈 希 表 ， 
用 来 统计 字符 串 中 每 个 字符 出 现 的 次 数 。 当 扫描 到 第 一 个 字符 串 中 的 每 
个 字符 时 ， 为 哈 希 表 对 应 的 项 的 值 增加 1。 接 下 来 扫描 第 二 个 字符 串 ， 
扫描 到 每 个 字符 时 ， 为 哈 希 表 对 应 的 项 的 值 减 去 1。 如 果 扫 摘 完 第 二 个 
ae 哈 希 表 中 所 有 的 值 都 是 0， 那 么 这 两 个 字符 串 就 互 为 变 位 
词 。 
































举一反三 : 


如 果 需 要 判断 多 个 字符 是 不 是 在 茶 个 字符 串 里 出 现 过 或 者 统计 多 个 字符 在 某 个 字符 串 中 
出 现 的 次 数 ， 我 们 可 以 考虑 基于 数组 创建 一 个 简单 的 哈 希 表 。 这 样 可 以 用 很 小 的 空间 消耗 换 来 
























































时 间 效率 的 提升 。 
面试 题 36: 数组 中 的 逆序 对 


题目 : 在 数组 中 的 两 个 数字 如 果 前 面 一 个 数字 大 于 后 面 的 
数字 ， 则 这 两 个 数字 组 成 一 个 逆序 对 。 输 入 一 个 数组 ， 求 出 这 
个 数组 中 的 逆序 对 的 总 数 。 

例如 在 数组 {7,5,6,4} 中 ， 一 共存 在 5 个 逆序 对 ， 分 别 是 (7,6) 、 

(7.5) = Chay: "C6 4) C5 AY. 

看 到 这 个 题目 ， 我 们 的 第 一 反应 是 顺序 扫描 整个 数组 。 每 扫描 到 一 
个 数字 的 时 候 ， 逐 个 比较 该 数字 和 它 后 面 的 数字 的 大 小 。 如 果 后 面 的 数 
字 比 它 小 ， 则 这 两 个 数字 就 组 成 了 一 个 逆序 对 。 假 设 数组 中 含有 n 个 数 
字 。 由 于 每 个 数字 都 要 和 O (Cn) 个 数字 作 比 较 ， 因 此 这 个 算法 的 时 间 
复杂 上 度 是 O w) 。 我 们 再 尝试 找 找 更 快 的 算法 。 

我 们 以 数组 {17,5,6,4} 为 例 来 分 析 统 计 逆 序 对 的 过 程 。 每 次 扫 摘 到 一 
个 数字 的 时 候 ， 我 们 不 能 拿 它 和 后 面 的 每 一 个 数字 作 比 较 ， 人 否则 时 间 复 
林 度 就 是 O Cn) ， 因 此 我 们 可 以 考虑 先 比 较 两 个 相 邻 的 数字 。 

如 图 5.1 Ca) 和 图 5.1 b) 所 示 ， 我 们 先 把 数组 分 解 成 两 个 长 度 为 2 
的 子 数组 ， 再 把 这 两 个 子 数 组 分 别 拆 分 成 两 个 长 度 为 1 的 子 数 组 。 接 下 
来 一 边 合 并 相 邻 的 子 数 组 ， 一 边 统 计 逆 序 对 的 数目 。 在 第 一 对 长 度 为 1 
的 子 数组 {7}、{5} 中 7 大 于 5， 因 此 (7,5) 组 成 一 个 逆序 对 。 同 样 在 第 二 
对 长 度 为 1 的 子 数组 {6}、{4} 中 也 有 逆序 对 (6,4) 。 由 于 我 们 已 经 统计 
了 这 两 对 子 数组 内 部 的 逆序 对 ， 因 此 需要 把 这 两 对 子 数组 排序 (图 
5.1 (c) 所 示 ) ， 以 免 在 以 后 的 统计 过 程 中 再 重复 统计 。 














Ca) 把 长 度 为 4 的 数组 分 解 成 两 个 长 度 为 2 的 子 数组 


(b) 把 长 度 为 2 的 数组 分 解 成 两 个 长 度 为 1 的 子 数组 


C) 把 长 度 为 1 的 子 数组 合并 、 排 序 ， 并 统计 逆序 对 


(d) 把 长 度 为 2 的 子 数组 合并 、 排 序 ， 并 统计 逆序 对 








图 5.1 统计 数组 {7,5,6,4} 中 逆序 对 的 过 程 


图 中 省 略 了 最 后 一 步 ， 即 复制 第 二 个 子 数 组 最 后 剩余 的 4 到 辅助 数组 中 。 ee 
o EP OR RAC, eH BUA EEE. PQ AEE AN SR 
数字 ， 因 此 第 二 个 子 数组 中 有 两 个 数字 比 7 小 。 把 逆序 对 数目 加 2， 并 把 7 复制 到 辅助 数组 ， at 
移动 P1 和 P3。 (b) P1 指 向 的 数字 小 于 P2 指 向 的 数字 ， 没 有 逆序 对 。 把 P2 指 向 的 数字 复制 到 辅 
助 数组 ， 并 向 前 移动 P2 和 P3。 (o) P1 指 向 的 数字 大 于 P2 指 向 的 数字 ， 因 此 存在 逆序 对 。 由 于 
P2 指 向 的 数字 是 第 二 个 子 数组 的 第 一 个 数字 ， 子 数组 中 只 有 一 个 数字 比 5 小 。 把 逆序 对 数目 加 
1， 并 把 5 复制 到 辅助 数组 ， 向 前 移动 P1 和 P3。 


接 下 来 我 们 统计 两 个 长 度 为 2 的 子 数组 之 间 的 逆序 对 。 我 们 在 图 5.2 
pm a T 
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(a) (b> 


图 5.2 ”图 5.1 Cd) 中 合并 两 个 子 数组 并 统计 逆序 对 的 过 程 


我 们 先 用 两 个 指针 分 别 指 癌 两 个 子 数 组 的 末尾 ， 并 每 次 比较 两 个 指 
针 指向 的 数字 。 如 果 第 一 个 子 数组 中 的 数字 大 于 第 二 个 子 数 组 中 的 数 
字 ， 则 构成 逆序 对 ， 并 且 逆 序 对 的 数目 等 于 第 二 个 子 数 组 中 剩余 数字 的 
个 数 〈 如 图 5.2〈a) 和 图 5.2《〈c) 所 示 ) 。 如 果 第 一 个 数组 中 的 数字 小 
于 或 等 于 第 二 个 数组 中 的 数字 ， 则 不 构成 逆序 对 (如 图 5.2(b) 所 
示 ) 。 每 一 次 比较 的 时 候 ， 我 们 都 把 较 大 的 数字 从 后 往 前 复制 到 一 个 畏 
助 数组 中 去 ， 确 保 辅助 数组 中 的 数字 是 递增 排序 的 。 在 把 较 大 的 数字 复 
制 到 辅助 数组 之 后 ， 把 对 应 的 指针 问 前 移动 一 位 ， 接 下 来 进行 下 一 轮 比 


较 。 

经 过 前 面 详 细 的 讨论 ， 我 们 可 以 总 结 出 统计 逆序 对 的 过 程 : 先 把 数 
组 分 隔 成 子 数 组 ， 先 统计 出 子 数组 内 部 的 逆序 对 的 数目 ， 然 后 再 统计 出 
两 个 相 邻 子 数 组 之 间 的 首 序 对 的 数目 。 在 统计 逆序 对 的 过 程 中 ， 还 需要 
对 数组 进行 排序 。 如 果 对 排序 算法 很 熟悉 ， 我 们 不 难 发 现 这 个 排序 的 过 
程 实际 上 就 是 归并 排序 。 我 们 可 以 基于 归并 排序 写 出 如 下 代码 : 











int InversePairs(int* data, int length) 


{ 
if(data == NULL || length < 0) 


return 0; 


int* copy = new int[length]; 
for(int i OF 1 :< Length? ++ ił) 
copy[i] = data[i]; 


int count = InversePairsCore(data, copy, 0, length - 1); 
delete[] copy; 


return count; 


int InversePairsCore(int* data, int* copy, int start, int end) 
{ 
if(start == end) 


{ 
copy [start] = data[start]; 


return 0; 


} 


} 


int length = (end - start) / 2; 
int left = InversePairsCore(copy, data, start, start + length); 
int right = InversePairsCore (copy, data, start + length + 1, end); 
// i 初始 化 为 前 半身 最 后 一 个 数字 的 下 标 
int i = start + length; 
// j 初始 化 为 后 半 段 最 后 一 个 数字 的 下 标 
int j = end; 
int indexCopy = end; 
int count = 0; 
while(i >= start && J] >= start + length + 1) 
{ 
if(data[i] > data[j]) 
{ 
copy[indexCopy--] = data[i--]; 
count += j - start - length; 


} 
else 
{ 


copy[indexCopy--] = data[j--]; 


for(; i >= start; --i) 
copy[indexCopy--] = data[i]; 


for(; j >= start + length + 1; --j) 


copy[indexCopy--] = data[j]; 
return left + right + count; 


我 们 知道 归并 排序 的 时 间 复 杂 度 是 O Cnlogn) ， 比 最 直观 的 


O Cr) 要 快 ， 但 同时 归并 排序 需要 一 个 长 度 为 n 的 辅助 数组 ， 相 当 于 我 
AHO Cn) 的 空间 消耗 换 来 了 时 间 效 率 的 提升 ， 因 此 这 是 一 种 用 空间 
换 时 间 的 算法 。 


源 代码 : 


本 题 完 整 的 源 代 码 详 见 36_InversePairs 项 目 。 
测试 用 例 : 


e 功能 测试 (输入 未 经 排序 的 数组 、 递 增 排序 的 数组 、 递 减 排 序 
的 数组 ， 输 入 的 数组 中 包含 重复 的 数字 ) 。 
边界 值 测试 “输入 的 数组 中 只 有 两 个 数字 、 数 组 的 数组 只 有 一 
SBF) 

e 特殊 输入 测试 〈 表 示 数 组 的 指针 为 NULL 指 针 ) 。 

本 题 考 点 : 

e 考查 分 析 复 杂 问 题 的 能 力 。 统 计 逆 序 对 的 过 程 很 复杂 ， 如 何 发 
现 逆序 对 的 规律 ， 是 应 聘 者 解决 这 个 题目 的 关键 。 

e 考查 应 聘 者 对 归并 排序 的 掌握 程度 。 如 果 应 聘 者 在 分 析 统 计 逆 
序 对 的 过 程 中 发 现 问 题 与 归并 排序 的 相似 性 ， 并 能 基于 归并 排序 形成 解 
题 思路 ， 那 通过 这 轮 面 试 的 几率 束 很 高 了 。 


面试 题 37: 两 个 链表 的 第 一 个 公共 结 点 
题目 : 输入 两 个 链表 ， 找 出 它们 的 第 一 个 公共 结 点 。 链 表 
结 点 定义 如 下 : 


struct ListNode 


f 
1 











int m_nKey; 
ListNode* m_pNext; 


TAY EN (ee le BUT EL, (RS ES A — BM ER IE: 在 第 
一 链表 上 顺序 壳 历 每 个 结 点 ， 每 壳 历 到 一 个 结 点 的 时 候 ， 在 第 二 个 链表 
上 顺序 遍历 每 个 结 点 。 如 果 在 第 二 个 链表 上 有 一 个 结 点 和 第 一 个 链表 上 
的 结 点 一 样 ， 说 明 两 个 链表 在 这 个 结 点 上 重合 ， 于 是 束 找 到 了 它们 的 公 
共 结 点 。 如 果 第 一 个 链表 的 长 度 为 mn， 第 二 个 链表 的 长 度 为 n"， 显 然 该 
方法 的 时 间 复 杂 度 是 O (m) 。 

通 和 覃 力 法 不 会 是 最 好 的 办 法 ， 我 们 接 下 来 试 着 分 析 有 公共 结 点 的 
两 个 链表 有 哪些 特点 。 从 链表 结 点 的 定义 可 以 看 出 ， 这 两 个 链表 是 单 回 
链表 。 如 果 两 个 单 问 链表 有 公共 的 结 点 ， 那 么 这 两 个 链表 从 某 一 结 点 开 
始 ， 它 们 的 m_pNext 都 指 问 同一 个 结 点 。 但 由 于 是 单 问 链表 的 结 点 ， 每 
个 结 点 只 有 一 个 m_pNext， 因 此 从 第 一 个 公共 结 点 开始 ， 之 后 它们 所 有 
结 点 都 是 重合 的 ， 不 可 能 再 出 现 分 又 。 所 以 两 个 有 公共 结 点 而 部 分 重合 


F; 











的 链表 ， 拓 扑 形状 看 起 来 像 一 个 Y， 而 不 可 能 像 X (如 图 5.3 所 示 )〉。 








图 5.3 ”两 个 链表 在 值 为 6 的 结 点 处 交汇 


经 过 分 析 我 们 发 现 ， 如 果 两 个 链表 有 公共 结 点 ， 那 么 公共 结 点 出 现 
在 两 个 链表 的 尾部 。 如 果 我 们 从 两 个 链表 的 尾部 开始 往 前 比较 ， 最 后 一 
个 相同 的 结 点 就 是 我 们 要 找 的 结 点 。 可 问题 是 在 单 同 链表 中 ， 我 们 只 能 
从 头 结 点 开始 按 顺 友 裔 历 ， 最 后 才能 到 达 尾 结 点 。 最 后 到 达 的 尾 结 点 却 
要 最 先 被 比较 ， 这 听 起 来 是 不 是 像 “ 后 进 移出 ?? 于 是 我 们 就 能 想到 用 栈 
的 特点 来 解决 这 个 问题 : 分 别 把 两 个 链表 的 结 点 放 入 两 个 栈 里 ， 这 样 两 
个 链表 的 尾 结 扣 束 位 于 两 个 栈 的 栈 项 ， 接 下 来 比较 两 个 栈 项 的 结 点 是否 
相同 。 如 果 相 同 ， 则 把 栈 顶 弹出 接着 比较 下 一 个 栈 项 ， 直 到 找到 最 后 一 
个 相同 的 结 点 。 

在 上 述 思 路 中 ， 我 们 需要 用 两 个 辅助 栈 。 如 果 链 表 的 长 度 分 别 为 m 
和 n， 那 么 空间 复杂 度 是 O (mtn) 。 这 种 思路 的 时 间 复 杂 度 也 是 O_ (m 
+n) 。 和 最 开始 的 蛋 力 法 相 比 ， 时 间 效 率 得 到 了 提高 ， 相 当 于 是 用 空 
间 消 耗 换 取 了 时 间 效 率 。 

之 所 以 需要 用 到 栈 ， 是 因为 我 们 想 同时 遍历 到 达 两 个 栈 的 尾 结 点 。 
当 两 个 链表 的 长 度 不 相同 时 ， 如 果 我 们 从 头 开 始 轴 历 到 达 尾 结 点 的 时 间 
就 不 一 致 。 其 实 解决 这 个 问题 还 有 一 个 更 简单 的 办 法 : 首先 过 历 两 个 链 
表 得 到 它们 的 长 上 度 ， 残 能 知道 哪个 链表 比较 长 ， 以 及 长 的 链表 比 短 的 链 
表 多 几 个 结 皮 。 在 第 二 次 角 历 的 时 候 ， 在 较 长 的 链表 上 先 走 硝 干 步 ， 接 
大 再 同时 在 两 个 链表 上 和 遍历， 找到 的 第 一 个 相同 的 结 点 就 是 它们 的 第 一 
个 公共 结 点 。 

比如 在 图 5.3 的 两 个 链表 中 ， 我 们 可 以 匈 过 历 一 次 得 到 它们 的 长 度 
分 别 为 5 和 4， 也 就 是 较 长 的 链表 与 较 短 的 链表 相 比 多 一 个 结 点 。 第 二 次 
先 在 长 的 链表 上 走 1 步 ， 到 达 结 点 2。 接 下 来 分 别 从 结 点 2 和 结 点 4 出 发 同 
时 所 历 两 个 结 点， 直到 找到 它们 第 一 个 相同 的 结 点 6， 这 就 是 我 们 想 要 


第 三 种 思路 和 第 二 种 思路 相 比 ， 时 间 复杂 度 都 是 0 (m+n) ， 但 我 
们 不 再 需要 辅助 的 栈 ， 因 此 提高 了 空间 效率 。 当 面试 官 首肯 了 我 们 节 后 
一 种 思路 之 后 ， 就 可 以 动手 写 代 码 了 。 下 面 是 一 段 参 考 代 码 : 























ListNode* FindFirstCommonNode( ListNode *pHeadl, ListNode *pHead2) 
{ 

// 得 到 两 个 链表 的 长 度 

unsigned int nLengthl GetListLength (pHead1) ; 

unsigned int nLength2 = GetListLength (pHead2) ; 

int nLengthDif = nLengthl - nLength2; 


ListNode* pListHeadLong = pHead1; 
ListNode* pListHeadShort = pHead2; 
if(nLength2 > nLength1) 
{ 
pListHeadLong = pHead2; 
pListHeadShort = pHeadi; 
nLengthDif = nLength2 - nLengthl; 
} 


// 先 在 长 链表 上 走 几 步 ， 再 同时 在 两 个 链表 上 遍历 
for(int i = 0; i < nLengthDif; ++ i) 
pListHeadLong = pListHeadLong->m_pNext; 


while((pListHeadLong != NULL) && 
(pListHeadShort != NULL) && 
(pListHeadLong != pListHeadShort) ) 


pListHeadLong = pListHeadLong->m_pNext; 
pListHeadShort = pListHeadShort-—>m_pNext; 
} 


// 得 到 第 一 个 公共 结 点 
ListNode* pFisrtCommonNode = pListHeadLong; 


return pFisrtCommonNode; 


} 


unsigned int GetListLength(ListNode* pHead) 
{ 
unsigned int nLength = 0; 
ListNode* pNode = pHead; 
while (pNode != NULL) 
{ 
++ nLength; 
pNode = pNode->m_pNext; 
} 


return nLength; 


源 代码 : 
本 题 完整 的 源 代码 详 见 37_FirstCommonNodesInLists 项 目 。 
测试 用 例 : 


o 功能 测试 (输入 的 两 个 链表 有 公共 交点 第 一 个 公共 结 反 在 链 
表 的 中 间 ， 第 一 个 公共 结 点 在 链表 的 末尾 ， 第 一 个 公共 结 扩 是 链表 的 头 
结 点 ; 输入 的 两 个 链表 没有 公共 结 皮 )。 

e 特殊 输入 测试 〈 输 入 的 链表 头 结 点 是 NULL 指 针 ) 


本 题 考点 : 


@ 考 香 应聘 者 对 时 间 复 杂 度 和 空间 复杂 上 度 的 理解 及 分 析 能 力 。 解 
决 这 道 题 有 多 种 不 同 的 思路 。 每 当 应 聘 者 想到 一 种 思路 的 时 候 ， 都 要 很 
人 
1 地方。 

e 考查 应 聘 者 对 链表 的 编程 能 


相关 题目 : 


如 果 把 图 5.3 逆 时 针 旋 转 90"， 我 们 就 会 发 现 两 个 链表 的 拓扑 形状 和 
一 柠 树 的 形状 非常 相似 ， 只 是 这 里 的 指针 是 从 叶 结 点 指 回 根 结 点 的 。 两 
个 链表 的 第 一 个 公共 结 点 正好 就 是 二 叉 树 中 两 个 叶 节 点 的 最 低 公共 祖 
先 。 在 本 书 7.2 节 ， 我 们 将 详细 讨论 如 何 求 两 个 结 点 的 最 低 公共 祖先 。 


54 ”本章 小 结 


编程 面试 的 时 候 ， 面 试 官 通常 对 时 间 复 杂 度 和 空间 复杂 度 都 会 有 要 
求 ， 并 且 一 般 情 况 下 面试 官 更 加 关注 时 间 复 杂 上 度 。 

降低 时 间 复 共度 的 第 一 个 方法 是 改 用 更 加 蜗 效 的 算法 。 比 如 我 们 用 
动态 规划 解答 面试 题 31“ 连 续 子 数组 的 最 大 和 ”能 够 把 时 间 复 洒 度 降低 到 
O(n) ， 利 用 快速 排序 的 Partition 函 数 也 能 在 O Cn) 时 间 解 决 面试 题 
29“ 数 组 中 出 现 次 数 超过 一 半 的 数字 ”和 面试 题 30“ 最 小 的 k 个 数字 ”。 

降低 时 间 复 杂 度 的 第 二 个 方法 是 用 空间 换取 时 间 。 在 解决 面试 题 
35“ 第 一 个 只 出 现 一 次 的 字符 ”的 时 候 ， 我 们 用 数组 实现 一 个 简单 的 哈 希 
表 ， 于 是 用 O (1) 时 间 就 能 知道 任意 字符 出 现 的 次 数 。 这 种 思路 可 以 
解决 很 多 同类 型 的 题目 。 另 外 ， 我 们 可 以 创建 一 个 缓存 保存 中 间 的 计算 
结果 ， 从 而 避免 重 复 的 计算 。 面 试题 34“ 丑 数 ” 束 是 这 方面 的 一 个 例子 。 









































在 用 递归 的 思路 求解 问题 的 时 候 ， 如 果 有 重复 的 子 问题 ， 同 样 我 们 也 可 
以 通过 保存 求解 子 问 题 的 结果 来 避免 重复 计算 。 更 多 关于 递归 的 讨论 请 
参考 本 书 的 2.4.2 及 面试 题 % 裴 波 那 外 数 列 ”。 

值得 注意 的 是 ， 以 空间 换取 时 间 并 不 一 定 都 是 可 行 的 方案 。 我 们 要 
注意 需要 的 辅助 空间 的 大 小 ， 消 耗 太 多 的 内 存 可 能 得 不 偿 失 。 为 外 ， 我 
们 还 要 关注 问题 的 背景 。 如 末 面 试题 是 有 关 稻 入 式 开 发 的 ， 那 对 空间 消 
耗 就 要 格外 留心 ， 因 为 通常 钥 入 式 系 统 的 内 存 很 有 限 。 








第 6 半 ”面试 中 的 各 项 能 力 
6.1 面试 官 谈 能 


“应 聘 者 能 够 礼貌 平和 、 不 皇 不 亢 地 和 面试 官 交流 ， 逻 辑 清晰 、 详 略 得 当地 介绍 自己 及 项 
目 经 历 ， 谈 论题 目 时 能 够 发 现 问题 的 细 贡 并 向 面试 官 进行 询问 ， 这 些 都 是 比较 好 的 沟通 表现 。 
对 目 己 做 的 项 目 能 够 了 解 很 深入 、 对 面试 题 能 够 快速 寻找 解决 万 法 是 判断 应 聘 痢 学 习 能 力 的 一 
个 方法 。 这 两 个 能 力 都 很 重要 ， 基 本 能 够 起 到 一 票 否 决 的 作用 。 


一 一 段 焰 支付宝， 高 级 安全 测试 工程 师 ) 


“有 时 候 会 问 一 些 应 聘 者 不 是 很 熟悉 的 领域 ， 看 应 聘 者 在 过 到 难题 时 的 反应 ， 在 他 们 回答 
不 出 时 会 有 人 员 提 供 解 答 ， 在 解答 过 程 中 观察 他 的 沟通 能 力 及 求知 欲 。” 
一 一 朱 说 (交通 银行 ， 项 目 经 理 
“沟通 能 力 其 实 整个 过 程 都 在 考核 ， 包 括 询问 他 过 往 的 经 历 ， 也 通常 会 涉及 沟通 能 
习 能 力 是 在 考查 算法 或 者 项 目 经 验 过 程 中 ， 通 过 提问 ， TUE Hebe A A LAE eH 


的 。 沟 通 能 力 和 学 习 能 力 很 重要 ， 在 某 种 程度 上 这 些 都 是 潜力 。 如 果 应 聘 者 沟通 能 力 不 行 、 难 
以 合作 ， 我 们 不 会 录取 。” 
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可 幸 杰 〈SAP， 高 级 工程 师 ) 


“让 其 介绍 过 往 项 目 其 实 就 在 考查 沟通 和 表达 能 力 。 学 习 能 力 通过 问 其 看 书 和 关注 什么 来 
考查 。 沟 通 能 力 、 学 习 能 力 对 最 终 面 试 结果 会 有 一 定 的 影响 。 对 于 资深 的 应 聘 者 ， 影 响 要 大 



















































































一 一 韩 伟 东 〈 盛 大， 高 级 研究 员 ) 


“应 聘 者 会 被 问 及 一 些 需 求 不 是 很 明确 的 问题 ， 解 决 这 些 问 题 需要 应 聘 者 和 面试 官 进行 沟 
通 ， 以 及 在 讲解 设计 思路 和 代码 的 过 程 中 也 需要 和 面试 官 交 流 互 动 。 沟 通 及 学 习 能 力 是 面试 成 
绩 中 关键 的 考查 点 。” 




































































“沟通 、 学 习 能 力 就 是 看 面试 者 能 否 清晰 、 有 条 理 地 表达 自己 ， 古 否 会 在 自己 所 得 到 的 信 
县 不 够 的 情况 下 主动 发 问 澄清 ， 能 否 在 得 到 一 些 暗 示 之 后 迅速 做 出 反应 纠正 错误 。” 
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6.2 ”沟通 能 力 和 学 习 能 力 
沟通 能 力 
随 着 软件 、 系 统 功能 越 来 越 复杂 ， 开 发 团队 的 规模 也 随 之 扩张 ， 开 





发 者 、 测 试 者 和 项 目 经 理 之 间 的 沟通 交流 也 变 得 越 来 越 重要 。 也 正 因为 
此 ， 很 多 公司 在 面试 的 时 候 都 会 注意 考 仁 应 聘 者 的 沟通 能 力 。 这 束 要 求 
应 聘 者 无 论 是 在 介绍 项 目 经 验 还 是 介绍 解 题 思 路 的 时 候 ， 都 需要 逻辑 清 
晰 明了 ， 语 言 详 略 得 当 ， 表 述 的 时 候 重 点 突出 、 观 点 明确 。 

我 们 不 能 把 好 的 沟通 能 力 理 解 成 夺 夸 其 谈 。 在 面试 的 时 候 ， 知 之 为 
知之 ， 不 知 为 不 知 ， 对 于 不 清楚 的 知识 点 ， 要 勇敢 承认 ， 千 万 别 不 懂 装 
展 。 通 常 当 应 聘 者 说 自己 很 懂 某 一 领域 的 时 候 ， 面 试 官 都 会 跟 进 几 个 问 
题 。 如 采 应 聘 者 在 不 懂 装 懂 ， 面 试 官 迟早 会 发 现 ， 他 可 能 束 会 党 得 应 聘 
者 在 其 他 的 地 方 也 有 浮 报 虚 他 的 成 分 ， 这 将 是 得 不 偿 失 的 。 

有 意 癌 加 入 外 企 的 应 聘 者 要 注意 提高 目 己 英文 交流 的 能 力 。 不 少 外 
企 的 面试 部 分 甚至 全 部 采用 英语 面试 ， 这 对 英语 的 要 求 就 很 高 。 我 们 通 
过 了 瑞 语 的 四 六 级 考试 未 必 能 用 英语 对 话 。 如 条 和 觉得 目 己 英 语 的 听 说 能 
力 还 不 够 好 ， 建 议 伦 更 多 的 时 间 来 提高 目 己 的 听力 。 英 语 面试 中 最 重要 
的 是 我 们 要 听 懂 面试 官 的 问题 。 通 党 采用 英文 面试 的 面试 官 上 自己 的 英语 
都 比较 好 ， 即 使 我 们 的 发 音 不 够 标准 ， 对 方 一 般 也 能 听 懂 。 这 和 我 们 能 
听 懂 普通 话 不 标准 的 老外 说 中 文 的 道理 是 一 样 的 。 但 如 果 面 试 官 的 问题 
没有 听 明 白 ， 那 我 们 就 是 说 得 再 清楚 也 无 济 于 事 了 。 


2. 学 习 能 力 


计算 机 是 一 门 更 新 速度 很 快 的 学 科 ， 每 年 都 有 新 的 技术 不 断 涌现 。 
因此 作为 这 个 领域 从 业 人 员 的 软件 工程 师 们 需要 要 具备 很 强 的 学 习 能 
， 人 否则 时 间 一 长 融会 跟 不 上 技术 进步 的 步伐 。 也 正 是 因为 这 个 原因 ， 
IT 公司 在 面试 的 时 候 ， 面 试 官 都 会 重视 应 聘 者 的 学 习 能 力 。 只 有 具备 很 
强 的 学 习 能 力 及 学 习 愿 望 的 人 ， 才 能 不 断 完 善 目 己 的 知识 结构 ， 不 断 学 
习 新 的 先进 技术 ， 让 自己 的 职业 生涯 保持 长 久 的 生命 力 。 

通常 面试 官 有 两 种 办 法 考查 应 聘 者 的 学 习 能 力 。 第 一 种 方法 是 询问 
应 聘 者 最 近 在 看 什么 书 或 者 在 做 什么 项 目 、 从 中 学 到 了 哪些 新 技术 。 面 
试 官 可 以 用 这 个 问题 了 解 应 聘 者 的 学 习 愿 望 和 学 习 能 力 。 学 习 能 力 强 的 
人 对 各 种 新 技术 充满 了 兴趣 ， 随 时 学 习 、 吸 收 新 知识 ， 并 把 知识 转换 为 
目 己 的 技能 。 第 二 种 方法 是 抛 出 一 个 新 概念 ， 接 下 来 观察 应 聘 者 能 不 能 
在 较 短 时 间 内 理解 这 个 新 概念 并 解决 相关 的 问题 。 本 书 收集 的 面试 题 涉 
及 诸如 数组 的 旋转 〈 面 试题 8) 、 二 又 树 的 镜像 〈 面 试题 19) . A 

〈 面 试题 34) 、 逆 序 对 (面试 题 36) 等 新 概念 。 当 面试 官 提 出 这 些 新 概 
念 的 时 候 ， 他 期 待 应 聘 者 能 够 通过 思考 、 提 问 、 再 思考 的 过 程 ， 理 解 它 
们 并 最 终 解决 问题 。 
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力 。 学 习 能 力 上 怎么 体现 呢 ? 面 试 官 提出 一 个 新 概念 ， 应 聘 者 没有 听 说 过 
它 ， 于 是 他 在 已 有 的 理解 的 基础 上 提出 进一步 的 问题 ， 得 到 面试 官 的 答 
复 之 后 ， 思 考 再 提问 ， 几 个 来 回 之 后 掌握 了 这 个 概念 。 这 个 过 程 能 够 体 
现 应 聘 者 的 学 习 能 力 。 通 常 学 习 能 力 强 的 人 具有 主动 积极 的 态度 ， 对 未 
知 的 领域 有 强烈 的 求知 欲望 。 因 此 建议 应 聘 者 在 面试 过 程 中 直到 不 明白 
的 地 方 多 提问 ， 这 样 面试 官 就 会 党 得 你 态度 积极 、 求 知 欲望 强烈 ， 会 给 
面试 结果 加 分 。 

面试 小 提示 : 
面试 是 一 个 双向 交流 的 过 程 ， 面 试 官 可 以 问 应 聘 者 问题 ， 同 样 应 聘 者 也 可 以 向 面试 官 提 
问 。 如 果 应 聘 者 能 够 针对 面试 题 主 动 地 提出 几 个 高 质量 的 问题 ， 面 试 官 束 会 觉得 他 有 很 强 的 沟 
通 能 力 和 学 习 能 力 。 

举 个 例 于 ，Google 曾 经 有 一 道 面试 题 : 找 出 第 1500 个 丑 数 。 很 多 人 
都 不 知道 丑 数 是 什么 。 不 知道 怎么 办 ? 面试 官 就 坐 在 对 面 ， 可 以 问 他 。 
面试 官 会 告诉 你 只 含有 2、3、5 三 个 因子 的 数 就 是 丑 数 。 你 听 了 后 ， 筑 
得 听 明 白 了 ， 但 不 太 确 定 ， 于 是 可 以 举 几 个 例子 并 让 面试 官 确认 你 的 理 
解 是 不 是 正确 : 6、8、10、12 都 是 丑 数 ， 但 14 就 不 是 ， 对 吗 ? 当面 试 官 
给 出 肯定 的 答复 ， 你 就 知道 自己 的 理解 是 对 。 问 题 问 的 是 第 1500 个 丑 
数 ， 与 顺序 有 关 。 可 是 哪个 数字 是 第 一 个 丑 数 呢 ，1 和 是 不 是 第 一 个 ? 这 
个 你 可 能 也 不 能 确定 ， 怎 么 办 ? 还 是 问 面 试 官 ， 他 会 告诉 你 1 是 或 者 不 
征 丑 数 。 题 目 是 他 出 的 ， 他 有 责任 把 题目 解释 清楚 。 

有 些 面 试 官 故意 一 开始 不 把 题目 描述 清楚 ， 让 题目 存在 一 定 的 二 义 
性 。 他 期 竺 应 聘 者 能 够 一 步 步 通过 提问 来 弄 明日 题目 的 要 求 。 这 也 是 在 
考 奋 应 聘 者 的 沟通 能 力 。 为 什么 要 这 样 考查 ? 因为 实际 工作 也 是 这 样 ， 
不 是 一 开始 项 目 需 求 束 定义 得 很 清楚 ， 程 序 员 需 要 多 次 与 项 目 经 理 其 至 
客户 反复 沟通 才能 把 需求 并 清楚 。 如 果 疫 有 一 定 的 沟通 能 力 ， 当 程序 员 
面 对 一 个 模糊 的 客户 需求 时 他 束 会 觉得 无 从 下 手 。 

比如 最 近 很 流行 的 一 个 面试 题 ， 面 试 官 最 开始 问 : 如 何 求 树 中 两 个 
结 点 的 最 低 公 共 祖 先 。 此 时 面试 官 对 题目 中 的 树 的 特点 完全 没有 给 出 描 
述 ， 他 和 希望 应 聘 者 在 听 到 问题 后 会 提出 几 个 问题 ， 比 如 这 株 树 是 二 又 树 
还 是 普通 的 树 。 

如 果 面 试 官 说 是 二 叉 树 ， 应 聘 者 可 以 继续 问 该 树 是 不 是 排序 的 二 叉 
树 。 面 试 官 回答 是 排序 的 。 听 到 这 里 ， 应 聘 者 才能 确定 思路 : 从 树 的 根 
结 点 出 发 过 历 树 ， 如 果 当 前 结 反 都 大 于 输入 的 两 个 结 点 ， 则 下 一 步 吉 历 































































































当前 结 反 的 左 子 树 ， 如 果 当 前 结 点 小 于 输入 的 两 个 结 点 ， 则 下 一 步 这 历 
当前 结 点 的 右 子 树 。 一 直人 吉 历 到 当前 结 点 比 一 个 输入 结 点 大 而 比 男 一 个 
小 的 时 候 ， 此 时 当前 结 点 殊 是 符合 要 求 的 最 低 公 共和 祖先 。 

在 应 聘 者 问 树 是 不 是 二 又 树 的 时 候 如 果 面 试 官 回答 是 任意 的 树 ， 此 
时 应 聘 者 可 以 接着 提问 在 树 结 点 中 有 没有 指 问 父 结 皮 的 指针 。 如 果 面 试 
官 给 出 肯定 的 回答 ， 也 就 是 树 的 结 点 中 有 指 同 父 结 点 的 指针 ， 此 时 从 输 
入 的 结 点 出 发 ， 沿 着 指 同 父 结 点 的 指针 一 直到 树 的 根 结 点 ， 可 以 看 做 一 
个 链表 ， 因 此 这 个 题目 的 解法 融和 求 两 个 链表 的 第 一 个 公共 结 氮 的 解法 
古 一 样 的 了 。 如 果 面 试 官 给 出 的 是 否定 的 回答 ， 也 就 是 树 的 结 反 没有 指 
回 父 结 点 的 指针 ， 那 么 我 们 可 以 在 过 历 的 时 候 用 一 个 栈 来 保存 从 根 结 点 
到 当前 结 点 的 路 径 ， 最 终 把 它 转化 成 求 两 个 路 径 的 最 后 一 个 公共 结 点 。 
详细 的 解 题 过 程 请 参考 本 书 的 7.2 市 。 

面试 官 给 出 不 同 的 条 件 ， 这 将 是 3 个 完全 不 一 样 的 题目 。 如 果 一 开 
始 应 聘 者 没有 弄 清 楚 和 面试 官 的 意图 束 贸 然 动 手 解 题 ， 那 结果 很 有 可 能 是 
离 题 干 里 。 从 中 我 们 也 可 以 看 出 在 面试 过 程 中 沟通 的 重要 性 。 当 觉得 题 
目的 条 件 、 要 求 不 够 明确 的 时 候 ， 我 们 一 定 要 多 提问 以 消除 目 己 的 疑 
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6.3 ”知识 迁移 能 力 


所 谓 学 习 能 力 ， 很 重要 的 一 点 就 是 根据 已 经 掌握 的 知识 、 技 术 ， 能 
够 迅速 学 习 、 理 解 新 的 技术 并 能 运用 到 实际 工作 中 去 。 大 部 分 新 的 技术 
都 不 是 凭空 产生 的 ， 而 是 在 已 有 技术 的 基础 上 发 展 起 来 的 。 这 就 要 求 我 
们 能 够 把 对 已 有 技术 的 理解 迁移 到 学 习 新 技术 的 过 程 中 去 ， 也 就 是 要 有 具 
备 很 强 的 知识 迁移 能 力 。 以 学 习 编程 语言 为 例 ， 如 果 全 面 理解 了 C++ 的 
面 回 对 象 的 思想 ， 那 么 学 习 下 一 门面 问 对 象 的 语言 Java 就 不 会 很 难 。 在 
深刻 理解 了 Java 的 垃圾 回收 机 制 之 后 ， 再 去 学 习 另 外 一 门 托管 语言 比如 
C#， 也 会 很 容易 。 

面试 官 考查 知识 迁移 能 力 的 一 个 方法 是 把 经 典 的 问题 稍 作 变换 。 这 
个 时 候 面 试 官 期 待 应 聘 者 能 够 找到 和 经 典 问题 的 联系 ， 并 从 中 受到 启发 
把 解决 经 典 问题 的 思路 迁移 过 来 解决 新 的 问题 。 比 如 如 果 遇 到 面试 题 
38“ 数 字 在 排序 数组 中 出 现 的 次 数 ”"， 我 们 看 到 “排序 数组 ”就 可 以 想到 二 
分 查找 算法 。 通 常 二 分 查找 算法 用 来 在 一 个 排序 数组 中 查找 一 个 数字 。 
我 们 可 以 把 二 分 查找 的 思想 迁移 过 来 稍 作 变 换 ， 用 二 分 查找 算法 在 排序 
OO 
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面试 官 考查 知识 迁移 能 力 的 另 一 个 方法 就 是 先 问 一 个 简单 的 问题 ， 






































在 应 聘 者 解答 完 这 个 简单 的 问题 之 后 再 退 问 一 个 相关 的 同时 难度 也 更 大 
的 问题 。 这 个 时 候 面 试 官 希望 应 聘 者 能 够 总 结 前 面 解决 简单 问题 的 经 
验 ， 把 前 面 的 思路 、 方 法 迁移 过 来 。 比 如 在 面试 题 40“ 数 组 中 只 出 现 一 
次 的 数字 ”中 ， 面 试 官 移 问 一 个 简单 的 问题 即 数组 中 只 有 一 个 数字 只 出 
现 一 次 的 情况 。 在 应 聘 者 想 出 用 寞 或 的 办 法 找到 这 个 只 出 现 一 次 的 数字 
之 后 ， 他 再 奶 问 如 果 数 组 中 有 两 个 数字 只 出 现 一 次 ， 该 怎么 找 出 这 两 个 
数字 ? 这 个 时 候 应 聘 者 要 从 前 面 的 思路 中 得 到 启发 : 既然 有 办 法 找到 数 
组 中 只 出 现 一 次 的 一 个 数字 ， 那 当 数 组 中 有 两 个 数字 只 出 现 一 次 的 时 
候 ， 我 们 可 以 把 整个 数组 一 分 为 二 ， 每 个 子 数 组 中 包含 一 个 只 出 现 一 次 
的 数字 ， 这 样 我 们 就 能 在 两 个 子 数 组 中 分 别 找到 那 两 个 只 出 现 一 次 的 数 
字 。 接 下 来 我 们 就 可 以 集中 精力 去 想 办 法 把 数组 一 分 为 二 ， 这 样 就 能 找 
到 解决 问题 的 穷 门 ， 整 个 题目 的 难度 系数 就 降低 了 不 少 。 

知识 迁移 能 力 的 一 种 通俗 的 说 法 是 “举一反三 ”的 能 力 。 我 们 在 去 面 
试 之 前 ， 通 第 都 会 看 一 些 经 典 的 面试 题 。 然 而 题目 总 是 做 不 完 的 ， 我 们 
不 可 能 把 所 有 的 面试 题 都 准备 一 帝 。 因 此 更 重要 的 是 每 做 一 道 面 试题 的 
时 候 ， 都 要 总 结 这 荐 题 的 解法 有 什么 特点 ， 有 哪些 思路 是 可 以 应 用 到 同 
类 型 的 题目 中 去 的 。 比 如 为 了 解决 面试 题 “ 翻 转 单词 顺序 ”， 我 们 先 翻 园 
整个 句子 的 所 有 字符 ， 再 分 别 翻 转 每 个 单词 中 的 字符 。 这 样 多 次 翻转 字 
符 的 思路 也 可 以 运用 到 面试 题 “ 左 旋转 字符 串 ” 中 《面试 题 42) 。 在 解决 
面试 题 28“ 字 符 串 的 排列 ”之 后 ， 我 们 发 现 “ 八 旦 后 问题 "其 实 归 根 到 的 就 
是 数组 的 排列 问题 。 本 书 中 很 多 半 市 在 分 析 了 一 道 题 之 后 ， 列 举 了 和 这 
道 题 相关 的 题目 ， 读 者 可 以 通过 分 析 这 些 题目 的 相关 性 来 提高 举一反三 


























面试 题 38: 数字 在 排序 数组 中 出 现 的 次 数 


题目 : 统计 一 个 数字 在 排序 数组 中 出 现 的 次 数 。 例 如 输入 
排序 数组 {1,2,3,3,3,3,4,5} 和 数字 3， 由 于 3 在 这 个 数组 中 出 现 了 
4 次 ， 因 此 输出 4。 


既然 输入 的 数组 是 排序 的 ， 那 么 我 们 很 自然 地 就 能 想到 用 二 分 查找 
算法 。 在 题目 给 出 的 例子 中 ， 我 们 可 以 先 用 二 分 查找 算法 找到 一 个 3。 
由 于 3 可 能 出 现 多 次 ， 因 此 我 们 找到 的 3 的 左右 两 边 可 能 都 有 3， 于 是 我 
们 在 找到 的 3 的 左右 两 边 顺 序 扫描 ， 分 别 找 出 第 一 个 3 和 最 后 一 个 3。 因 
为 要 碍 找 的 数字 在 长 度 为 n 的 数组 中 有 可 能 出 现 O Gn) 次 ， 所 以 顺序 扫 
描 的 时 间 复 杂 度 是 DO Cn) 。 因 此 这 种 算法 的 效率 和 直接 从 头 到 尾 顺 序 
扫描 整个 数组 统计 3 出 现 的 次 数 的 方法 是 一 样 的 。 显 然 ， 面 试 官 不 会 满 





意 这 个 算法 ， 他 会 提示 我 们 还 有 更 快 的 算法 。 

接 下 来 我 们 思考 如 何 更 好 地 利用 二 分 查找 算法 。 假 设 我 们 是 统计 数 
字 k 在 排序 数组 中 出 现 的 次 数 。 在 前 面 的 算法 中 时 间 主 要 消耗 在 如 何 确 
定 重 复出 现 的 数字 的 第 一 个 k 和 最 后 一 个 k 的 位 置 上 ， 有 没有 可 能 用 二 分 
查找 算法 直接 找到 第 一 个 k 及 最 后 一 个 k 呢 ? 

我 们 先 分 析 如 何 用 二 分 查找 算法 在 数组 中 找到 第 一 个 k。 二 分 查找 
算法 总 是 先 拿 数组 中 间 的 数字 和 k 作 比较 。 如 果 中 间 的 数字 比 k 大 ， 那 么 
k 只 有 可 能 出 现在 数组 的 前 半 段 ， 下 一 轮 我 们 只 在 数组 的 前 半 段 得 找 就 
可 以 了 。 如 果 中 间 的 数字 比 k 小 ， 那 么 k 只 有 可 能 出 现在 数组 的 后 半 段 ， 
下 一 轮 我 们 只 在 数组 的 后 半 段 得 找 就 可 以 了 。 如 果 中 间 的 数字 和 k 相 等 
E? 我 们 先 判断 这 个 数字 是 不 是 第 一 个 k。 如 采 位 于 中 间 数 字 的 前 面 一 
个 数字 不 是 k， 此 时 中 间 的 数字 刚好 就 是 第 一 个 k。 如 果 中 间 数 字 的 前 面 
一 个 数字 也 是 k， 也 就 是 说 第 一 个 k 肯 定 在 数组 的 前 半 段 ， 下 一 轮 我 们 仍 
然 需 要 在 数组 的 前 半 段 查找 。 
基于 这 个 思路 ， 我 们 可 以 很 容易 地 写 出 递归 的 代码 找到 排序 数组 中 
第 一 了 k: 
int GetFirstK(int* data, int length, int k, int start, int end) 
{ 

if(start > end) 

return -1; 











int middleIndex = (start + end) / 2; 
int middleData = data[middleIndex]; 


if (middleData == k) 
{ 
if ((middleIndex > 0 && data[middleIndex - 1] != k) 
|| middleIndex == 0) 


return middleIndex; 
else 
end = middleIndex 一 1; 
} 
else if (middleData > k) 
end = middleIndex - 1; 
else 
start = middleIndex + 1; 


return GetFirstK(data, length, k, start, end); 


在 函数 GetFirstK 中 ， 如 果 数 组 中 不 包含 数字 k， 那 么 返回 -1。 如 果 


数组 中 包含 至 少 一 个 k， 那 么 返回 第 一 个 k 在 数组 中 的 下 标 。 

我 们 可 以 用 同样 的 思路 在 排序 数组 中 找到 最 后 一 个 k。 如 果 中 间 数 
字 比 k 大 ， 那 么 k 只 能 出 现在 数组 的 前 半 段 。 如 果 中 间 数 字 比 k 小 ，k 束 只 
能 出 现在 数组 的 后 半 段 。 如 果 中 间 数 字 等 于 k 呢 ? 我 们 需要 判断 这 个 k 是 
不 是 最 后 一 个 k， 也 就 是 中 间 数 字 的 下 一 个 数字 是 不 是 也 等 于 k。 如 条 下 
一 个 数字 不 是 k， 则 中 间 数 字 束 是 最 后 一 个 kT 了; 否则 下 一 轮 我 们 还 是 要 
在 数组 的 后 半 段 中 去 查找 。 我 们 同样 可 以 基于 递归 写 出 如 下 代码 : 


int GetLastK(int* data, int length, int k, int start, int end) 











int middleIndex = (start + end) / 2 
int middleData = data[middleIndex]; 


if ((middleIndex < length - 1 && data[middleIndex + 1] != k) 
|| middleIndex == length - 1) 
return middleiIndex; 

else 


start = middleIndex + 1; 


else if(middleData < k) 
start = middleIndex + 1; 


end = middleIndex - 1; 


return GetLastK(data, length, k, start, end); 
} 

和 函数 GetFirstK 类 似 ， 如 果 数 组 中 不 包含 数字 k， 那 么 GetLastK 返 
回 -1; 否则 返回 最 后 一 个 k 在 数组 中 的 下 标 。 

和 分 别 找到 第 一 个 k 和 最 后 一 个 k 的 下 标 之 后 ， 我 们 就 能 计算 出 k 在 
数组 中 出 现 的 次 数 了 。 相 应 的 代码 如 下 : 





int GetNumberOfK(int* data, int length, int k) 


{ 


int number = 0; 

if(data != NULL && length > 0) 

{ 
int first = GetFirstK(data, length, k, 0, length - 1); 
int last = GetLastK(data, length, k, 0, length - 1); 


i1f(first > -l && last > -1) 
number = last = first + 1; 


return number; 

在 上 述 代码 中 ，GetFirstK 和 GetLastK 都 是 用 二 分 查找 法 在 数组 中 查 
找 一 个 合乎 要 求 的 数字 ， 它 们 的 时 间 复 杂 度 都 是 O (logn) ， 因 此 
GetNumberOfK 的 总 的 时 间 复 杂 上 度 也 只 有 OQO Cogn) 。 





源 代码 : 
本 题 完整 的 源 代码 详 见 38_NumberOfK 项 目 。 
测试 用 例 : 


e 功能 测试 〈 数 组 中 包含 得 找 的 数字 ， 数 组 中 没有 碍 找 的 数字 ， 
碍 找 的 数字 在 数组 中 出 现 一 次 /多 次 ) 。 
e 边界 值 测试 〈 碍 找 数 组 中 的 最 大 值 、 最 小 值 ， 数 组 中 只 有 一 个 


Wp. p> 


e 特殊 输入 测试 (表示 数组 的 指针 为 NULL 指 针 〉。 
本 题 考点 : 


© 考 但 应 聘 者 的 知识 迁移 能 力 。 我 们 都 知道 二 分 查找 算法 可 以 用 
来 在 排序 数组 中 查找 一 个 数字 。 应 聘 者 如 果 能 够 运用 知识 迁移 能 力 ， 把 
问题 转换 成 用 二 分 查找 算法 查找 重复 数字 的 第 一 个 和 最 后 一 个 ， 那 么 这 
个 问题 也 就 解决 了 一 大 半 。 

© 考 香 应 聘 者 对 二 分 查找 算法 的 理解 程度 。 这 道 题 实际 上 是 二 分 
查找 算法 的 加 强 版 。 只 有 对 二 分 查找 算法 有 看 深刻 的 理解 ， 应 聘 者 才 有 
可 能 解决 这 个 问题 。 




















面试 题 39: 二 又 树 的 深度 


题目 一 : 输入 一 柠 二 又 树 的 根 结 点 ， 求 该 树 的 深度 。 从 根 
结 点 到 叶 结 点 依次 经 过 的 结 点 〈 含 根 、 叶 结 点 ) 形成 树 的 一 条 
路 径 ， 最 长 路 径 的 长 度 为 树 的 深度 。 

二 又 树 的 结 反 定义 如 下 : 
struct BinaryTreeNode 
{ 

int m_nValue; 

BinaryTreeNode* m_pLeft; 

BinaryTreeNode* m_pRight; 


例如 ， 图 6.1 中 的 二 又 树 的 深度 为 4， 因 为 它 从 根 结 点 到 叶 结 点 最 长 


的 路 径 包 含 4 个 结 点 〈 从 根 结 点 1 开始 ， 经 过 结 点 2 和 结 点 5， 最 终 到 达 叶 
vo 


}; 








图 6.1 深度 为 4 的 三 又 树 








在 本 题 中 面试 官 给 出 了 一 种 树 的 深度 的 定义 ， 我 们 可 以 根据 这 个 定 
义 去 得 到 树 的 所有 路 径 ， 也 就 能 得 到 最 长 的 路 径 及 它 的 长 度 。 在 面试 题 
25“ 二 叉 树 中 和 为 茶 一 值 的 路 径 ” 中 我 们 详细 讨论 了 如 何 记 录 树 中 的 路 
径 。 这 种 思路 的 代码 量 比较 大 ， 我 们 可 以 尝试 更 加 简 清 的 方法 。 

我 们 还 可 以 从 忆 外 一 个 角度 来 理解 树 的 深度 。 如 果 一 棵 树 只 有 一 个 
结 点 ， 它 的 深度 为 1。 如 果 根 结 皮 只 有 左 子 树 而 没有 右 子 树 ， 那 么 树 的 
深度 应 该 是 其 左 子 树 的 深度 加 1;， 同样 如 果 根 结 点 只 有 右 子 树 而 没有 磊 
子 树 ， 那 么 树 的 深度 应 该 是 其 右 子 树 的 深度 加 1。 如 果 既 有 右 子 树 双 有 
左 子 树 ， 那 该 树 的 深度 就 是 其 左 、 右 子 树 深 度 的 较 大 值 再 加 1。 比 如 在 
图 6.1 的 二 又 树 中 ， 根 结 反 为 1 的 树 有 左右 两 个 子 树 ， 其 左右 子 树 的 根 结 





























点 分 别 为 结 点 2 和 3。 根 结 点 为 2 的 左 子 树 的 深度 为 93， 而 根 结 点 为 3 的 右 
子 树 的 深度 为 2， 因 此 根 结 点 为 1 的 树 的 深度 就 是 4。 

这 个 思路 用 递归 的 方法 很 容易 实现 ， 只 需 对 过 有 历 的 代码 稍 作 修改 即 
可 。 参 考 代码 如 下 : 


int TreeDepth (BinaryTreeNode* pRoot) 





if(pRoot == NULL) 
return 0; 


int nLeft = TreeDepth (pRoot->m_pLeft) ; 

int nRight = TreeDepth (pRoot->m_pRight) ; 

return (nLeft > nRight) ? (nLeft + 1) : (nRight + 1); 
} 

源 代码 : 

本 题 完整 的 源 代 码 详 见 39_1_TreeDepth 项 目 。 

测试 用 例 : 


ag. 功能 测试 〈 输 入 普通 的 二 叉 树 ， 二 又 树 中 所 有 结 点 都 没有 左 / 右 
对 ) 。 

e 特殊 输入 测试 “二叉树 只 有 一 个 结 点 ， 二 又 树 的 头绪 点 为 
NULL 指 针 ) 。 

只 要 应 聘 者 对 二 叉 树 这 一 数据 结构 很 熟悉 ， 束 能 很 快 写 出 上 面 的 代 
码 。 如 果 公 司 对 编程 能 力 有 较 高 的 要 求 ， 面 试 官 可 能 会 退 加 一 个 与 前 面 
问题 相关 但 难度 更 大 的 问题 。 比 如 ， 在 应 聘 者 做 完 上 面 的 问题 之 后 ， 面 
试 官 追问 : 

题目 二 : 输入 一 棵 二 叉 树 的 根 结 点 ， 判断 该 树 古 不 是 阅 衡 
二 叉 树 。 如 果 某 二 又 树 中 任意 结 点 的 左右 子 树 的 次 度 相差 不 超 
W1, MACHER a SOM. Bilin, Ale. Fay XA 
是 一 棵 平衡 二 叉 树 。 
需要 重复 遍历 结 点 多 次 的 解法 ， 简 单 但 不 足以 打动 面试 官 

有 了 求 二 叉 树 的 深度 的 经 验 之 后 再 解决 这 个 问题 ， 我 们 很 容易 就 能 


想到 一 个 思路 : 在 过 历 树 的 每 个 结 点 的 时 候 ， 调 用 函数 TreeDepth 得 到 
它 的 左右 子 树 的 深度 。 如 果 每 个 结 扣 的 左右 子 树 的 深度 相差 都 不 超过 














1， 按 照 定 义 它 就 是 一 棵 平衡 的 二 又 树 。 这 种 思路 对 应 的 代码 如 下 : 
bool IsBalanced(BinaryTreeNode* pRoot) 
{ 
if (pRoot == NULL) 
return true; 


int left = TreeDepth (pRoot->m_pLeft) ; 
int right = TreeDepth(pRoot->m_pRight) ; 
int diff = left: ~ right} 
EE (aarti & 1 [| dick sed} 

return false; 


return IsBalanced(pRoot->m_pLeft) && IsBalanced(pRoot->m_pRight) ; 


上 面 的 代码 固然 简洁 ， 但 我 们 也 要 注意 到 由 于 一 个 结 点 会 被 重复 过 
历 多 次 ， 这 种 思路 的 时 间 效 率 不 高 。 例 如 在 函数 IsBalance 中 输入 图 6.1 中 
的 和 二叉树 ， 我 们 将 首先 判断 根 结 点 ( 结 点 1) 是 不 是 平衡 的 。 此 时 我 们 
往 函 数 TreeDepth 输 入 左 子 树 的 根 结 点 〈 结 点 2) 时 ， 需 要 人 吉 历 结 点 4、 
5、7。 接 下 来 判断 以 结 点 2 为 根 结 点 的 子 树 是 不 是 平衡 树 的 时 候 ， 仍 然 
会 通 历 结 点 4、5、7。 训 无 疑问 ， 重 复 过 历 同 一 个 结 点 会 影 啊 性 能 。 接 
下 来 我 们 寻找 不 需要 重复 过 历 的 算法 。 


每 个 结 皮 只 避 历 一 次 的 解法 ， 正 古 面试 官 暑 欢 的 


如 果 我 们 用 后 序 遍 历 的 方式 遍历 二 又 树 的 每 一 个 结 点 ， 在 遍历 到 一 
个 结 点 之 前 我 们 就 已 经 台历 了 它 的 左右 子 树 。 只 要 在 吉 历 每 个 结 点 的 时 
候 记 录 它 的 深度 《〈 东 一 结 点 的 深度 等 于 它 到 叶 市 点 的 路 径 的 长 度 ) ， 我 
Do E T 














bool IsBalanceQ(BinaryTreeNode* pRoot, int* pDepth) 
{ 
if(pRoot == NULL) 
{ 
*pDepth = 0; 
return true; 


} 


i 


ant left, right; 
if (IsBalanced(pRoot->m_pLeft, &left) 
&& IsBalanced(pRoot->m_pRight, &right)) 
{ 
int diff = left = right; 
i1f(diff <= 1 && diff >= -1) 
{ 
*pDepth = 1+ (left > right ? left : right); 
return true; 


} 
i 
return false; 


我 们 只 需 给 上 面 的 函数 传 入 二 文 树 的 根 结 扣 及 一 个 表示 结 扩 深度 的 
整 型 变量 即 可 ; 
bool IsBalanced(BinaryTreeNode* pRoot) 


{ 


} 


int depth = 0; 
return IsBalanced(pRoot, &depth); 
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源 代码 : 
本 题 完 整 的 源 代码 详 见 39_2_BalancedBinaryTree 项 目 。 
测试 用 例 : 


e 功能 测试 (平衡 的 三 叉 树 ， 不 是 平衡 的 二 叉 树 ， 二 又 树 中 所 有 
结 点 都 没有 左 / 右 子 树 ) 。 


} 











e 特殊 输入 测试 “二叉树 中 只 有 一 个 结 点 ， 二 叉 树 的 头 结 点 为 
NULL 指 针 ) 。 


本 题 考点 : 


e 考查 对 二 又 树 的 理解 及 编程 能 力 。 这 两 个 题 的 解法 实际 都 只 是 
树 的 过 历 算 法 的 应 用 。 

© 考 奋 对 新 概念 的 学 习 能 力 。 面 试 官 提出 一 个 新 的 概念 即 树 的 深 
度 ， 这 就 要 求 我 们 在 较 短 的 时 间 内 理解 这 个 概念 并 解决 相关 的 问题 。 记 
是 一 种 审 见 的 面试 题 侍 。 能 在 较 短 时 间 扩 和 苇 握 、 理 解 新 概念 的 能 力 ， 就 


。 考查 知识 迁移 的 能 力 。 如 果 面试 官 先 问 如 何 求 二 叉 树 的 深度 ， 
再 问 如 何 判断 一 棵 二 又 树 是 不 是 平衡 的 ， 应 聘 者 应 该 从 求 二 又 树 深度 的 
分 析 过 程 中 得 到 启发 ， 找 到 判断 平衡 二 又 树 的 突破 口 。 


面试 题 40: 数组 中 只 出 现 一 次 的 数字 


题目 ; 一 个 整 型 数组 里 除了 两 个 数字 之 外 ， 其 他 的 数字 都 
出 现 了 两 次 。 请 写 程序 找 出 这 两 个 只 出 现 一 次 的 数字 。 要 求 时 
间 复 杂 度 是 O Cn) ， 空 间 复 杂 度 是 O (1) 。 

例如 输入 数组 {2,4,3,6,3,2,5,5}， 因 为 只 有 4、6 这 两 个 数字 只 出 现 一 
次 ， 其 他 数字 都 出 现 了 两 次 ， 所 以 输出 4 和 6。 

这 是 一 个 比较 难 的 题目 ， 很 少 有 人 能 在 面试 的 时 候 不 需要 提示 一 下 
子 想到 最 好 的 解决 办 法 。 一 般 当 应 聘 者 想 了 几 分 钟 后 还 没有 思路 ， 面 试 
官 会 给 出 一 些 提示 。 面 试 官 很 有 可 能 会 说 : 你 可 以 先 考 虑 这 个 数组 中 只 
有 一 个 数字 只 出 现 一 次 ， 其 他 的 都 出 现 了 两 次 ， 怎 么 找 出 这 个 数字 ? 

这 两 个 题目 都 在 强调 一 个 (或 两 个 ) 数字 只 出 现 一 次 ， 其 他 的 出 现 
两 次 。 这 有 什么 意义 呢 ? 我 们 想到 异 或 运算 的 一 个 性 质 :; 任何 一 个 数字 
异 或 它 自己 都 等 于 0。 也 就 是 说 ， 如 果 我 们 从 头 到 尾 依次 异 或 数组 中 的 
每 一 个 数字 ， 那 么 最 终 的 结果 刚好 是 那个 只 出 现 一 次 的 数字 ， 因 为 那些 
成 对 出 现 两 次 的 数字 全 部 在 异 或 中 抵消 了 。 

想 明 白 怎么 解决 这 个 简单 问题 之 后 ， 我 们 再 回 到 原始 的 问题 ， 看 看 
能 不 能 运用 相同 的 思路 。 我 们 试 着 把 原 数组 分 成 两 个 子 数组 ， 使 得 每 个 
子 数组 包含 一 个 只 出 现 一 次 的 数字 ， 而 其 他 数字 都 成 对 出 现 两 次 。 如 果 
能 够 这 样 拆 分 成 两 个 数组 ， 我 们 就 可 以 按照 前 面 的 办 法 分 别 找 出 两 个 只 
出 现 一 次 的 数字 了 。 

我 们 还 是 从 头 到 尾 依 次 异 或 数组 中 的 每 一 个 数字 ， 那 么 最 终 得 到 的 
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结 末 束 是 两 个 只 出 现 一 次 的 数字 的 异 或 结 末 。 因 为 其 他 数字 都 出 现 了 两 
次 ， 在 异 或 中 全 部 抵消 了 。 由 于 这 两 个 数字 肯定 不 一 样 ， 那 么 异 或 的 结 
果 肯 定 不 为 0， 也 就 是 说 在 这 个 结果 数字 的 二 进 制 表 示 中 人 至少 就 有 一 位 
为 1。 我 们 在 结果 数字 中 找到 第 一 个 为 1 的 位 的 位 置 ， 记 为 第 hn 位 。 现 在 
我 们 以 第 n 位 是 不 是 1 为 标准 把 原 数组 中 的 数字 分 成 两 个 子 数组 ， 第 一 个 
子 数组 中 每 个 数字 的 第 n 位 都 是 1， 而 第 二 个 子 数 组 中 每 个 数字 的 第 n 位 
都 是 0。 由 于 我 们 分 组 的 标准 是 数字 中 的 某 一 位 是 1 还 是 0， 那 么 出 现 了 
两 次 的 数字 肯定 被 分 配 到 同一 个 子 数组 。 因 为 两 个 相同 的 数字 的 任意 一 
位 都 是 相同 的 ， 我 们 不 可 能 把 两 个 相同 的 数字 分 配 到 两 个 子 数组 中 去 ， 
于 是 我 们 已 经 把 原 数 组 分 成 了 两 个 子 数组 ， 每 个 子 数组 都 包含 一 个 只 出 
现 一 次 的 数字 ， 而 其 他 数字 都 出 现 了 两 次 。 我 们 已 经 知道 如 何在 数组 中 
ee 


举 个 例子 ， 假 设 输入 数组 {2,4,3,6,3,2,5,5}。 当 我 们 依次 对 数组 中 的 
每 一 个 数字 做 异 或 运算 之 后 ， 得 到 的 结果 用 二 进 制 表 示 是 0010。 异 或 得 
到 结果 中 的 倒数 第 二 位 是 1， 于 是 我 们 根据 数字 的 倒数 第 二 位 是 不 是 1 分 
为 两 个 数组 。 第 一 个 子 数 组 {2,3,6,3,2} 中 所 有 数字 的 倒数 第 二 位 都 是 1， 
而 第 二 个 子 数组 {4,5,5} 中 所 有 数字 的 倒数 第 二 位 都 是 0(。 接 下 来 只 要 分 
别 对 这 两 个 子 数组 求 异 或 ， 就 能 找 出 第 一 个 子 数组 中 只 出 现 一 次 的 数字 
是 6， 而 第 二 个 子 数 组 中 只 出 现 一 次 的 数字 是 4。 

想 清 楚 整 个 过 程 之 后 再 写 代 码 就 不 难 了 。 下 面 是 参考 代码 : 














void FindNumsAppearOnce (int data[], int length, int* numl, int* num2) 
{ 
if (data == NULL || length < 2) 
return; 


int resultExclusiveOR = 0; 
for (int i = 0; i < length; ++ i) 
resultExclusiveOR ^= data[i]; 


unsigned int indexOfl = FindFirstBitIsl1 (resultExclusiveoOR) ; 


knuml = *num2 = 0; 
for (int j = 0; j< length; ++ 3) 
{ 
if (IsBit1(data[j], indexOf1) ) 
*numl ^= data[j]; 
else 
xnum2 ^= data[j]; 


} 


unsigned int FindFirstBitIsl(int num) 
{ 
int indexBit = 0; 
while (((num & 1) == 0) && (indexBit < 8 * sizeof(int))) 
{ 
num = num >> 1; 
++ indexBit; 


} 


return indexBit; 


} 


bool IsBitl(int num, unsigned int indexBit) 
{ 

num = num >> indexBit; 

return (num & 1); 


在 上 述 代 码 中 ，FindFirstBitIs1 用 来 在 整数 num 的 二 进 制 表示 中 找到 
最 右边 是 1 的 位 ，IsBit1 的 作用 是 判断 在 num 的 二 进 制 表示 中 从 右边 数 起 


的 indexBit 位 是 不 是 1。 


} 


源 代码 : 

本 题 完整 的 源 代码 详 见 40_NumbersAppearOnce 项 目 。 
测试 用 例 : 

功能 测试 〈 数 组 中 多 对 重复 的 数字 ， 数 组 中 没有 重复 的 数字 ) 
本 题 考点 : 


© 考 俘 知 识 了 迁移 能 力 。 只 有 一 个 数字 出 现 一 次 这 个 简单 的 问题 ， 
很 多 应 聘 者 都 能 想到 解决 办 法 。 能 不 能 把 解决 简单 问题 的 思路 迁移 到 复 
杂 问 题 上 ， 是 应 聘 者 能 否 通 过 这 轮 面 试 的 关键 。 

o 考 香 对 二 进 制 和 位 运算 的 理解 。 


面试 题 41: 和 为 s 的 两 个 数字 VS 和 为 s 的 连续 正 数 
序列 


题目 一 : 输入 一 个 递增 排序 的 数组 和 一 个 数字 s， 在 数组 中 
查找 两 个 数 ， 使 得 它们 的 和 正好 是 s。 如 果 有 多 对 数字 的 和 等 
于 s， 输 出 任意 一 对 即 可 。 

例如 输入 数组 {1、2、4、7、11、15} 和 数字 15。 由 于 4 十 11 二 15， 
因此 输出 4 和 11。 

面试 的 时 候 ， 很 重要 的 一 点 是 应 聘 者 要 表现 出 很 快 的 反应 能 力 。 只 
要 想到 一 个 方法 ， 应 聘 者 就 可 以 马上 告诉 面试 官 ， 即 使 这 个 方法 不 一 定 
是 最 好 的 。 比 如 这 个 问题 ， 很 多 人 都 能 立即 想到 O (mr) 的 方法 ， 也 就 
是 先 在 数组 中 国定 一 个 数字 ， 再 依次 判断 数组 中 其 余 的 n 一 1 个 数字 与 它 
的 和 是 不 是 等 于 s。 面 试 官 会 告诉 我 们 这 不 是 最 好 的 办 法 。 不 过 这 没有 
关系 ， 至 少 面试 官 知道 我 们 的 思维 还 是 比较 敏捷 的 。 

接着 我 们 寻找 更 好 的 算法 。 我 们 先 在 数组 中 选择 两 个 数字 ， 如 果 它 
们 的 和 等 于 输入 的 9s， 我 们 就 找到 了 要 找 的 两 个 数字 。 如 果 和 小 于 s 呢 ? 
我 们 希望 两 个 数字 的 和 再 大 一 点 。 由 于 数组 已 经 排 好 序 了 ， 我 们 可 以 考 
虑 选择 较 小 的 数字 后 面 的 数字 。 因 为 排 在 后 面 的 数字 要 大 一 些 ， 那 么 两 
个 数字 的 和 也 要 大 一 些 ， 就 有 可 能 等 于 输入 的 数字 s 了。 同样 ， 当 两 个 
数字 的 和 大 于 输入 的 数字 的 时 候 ， 我 们 可 以 选择 较 大 数字 前 面 的 数字 ， 
因为 排 在 数组 前 面 的 数字 要 小 一 些 。 

我 们 以 数组 {1、2、4、7、11、15} 及 期 待 的 和 15 为 例 详细 分 析 一 下 
这 个 过 程 。 首 先 定义 两 个 指针 ， 第 一 个 指针 指向 数组 的 第 一 个 (也 是 最 














小 的 ) 数字 1， 第 二 个 指针 指向 数组 的 最 后 一 个 〈 也 是 最 大 的 ) 数字 
15。 这 两 个 数字 的 和 16 大 于 15， 因 此 我 们 把 第 二 个 指针 向 前 移动 一 个 数 





字 ， 让 它 指 同 11。 这 个 时 候 两 个 数字 1 与 11 的 和 是 12， 小 于 15。 接 下 来 
我 们 把 第 一 个 指针 辣 后 移动 一 个 数字 指 问 2。 此 时 两 个 数字 2 与 11 的 和 
13， 还 是 小 于 15。 我 们 再 一 次 问 后 移动 第 一 个 指针 ， 让 它 指向 数字 4。 
数字 4、11 的 和 是 15， 正 是 我 们 期 竺 的 结果 。 表 6.1 总 结 了 在 数组 {1、 
2、4、7、11、15} 中 查找 和 为 15 的 数 对 的 过 程 。 


表 6.1 在 数组 {1、2、4、7、11、15} 中 查找 和 为 15 的 数 对 


较 小 的 数字 | 较 大 的 数字 和 与 s 相 比较 | 下 一 步 操作 
a h hb ë hk 


i 
| 
e ee | eee 


这 一 次 面试 官 会 首肯 我 们 的 思路 ， 于 是 就 可 以 动手 写 代 码 了 。 下 面 
是 一 段 参考 代码 : 

















bool FindNumbersWithSum(int data[], int length, int sum, 
int* numi, int* num2) 
{ 
bool found = false; 
if (length < 1 || numi == NULL || num2 == NULL) 
return found; 


int ahead = length - 1; 
int behind = 0; 


while (ahead > behind) 


{ 


long long curSum = data[ahead] + data[behind]; 
if(curSum == sum) 
{ 

*numl = data[behind]; 

*num2 = datal[ahead]; 

found = true; 

break; 


} 

else if(curSum > sum) 
ahead 一 一 ; 

else 
behind ++; 


} 


return found; 


在 上 述 代 码 中 ，ahead 为 较 小 的 数字 的 下 标 ，behind 为 较 大 的 数字 的 
下 标 。 由 于 数组 是 排序 的 ， 因 此 较 小 数字 一 定位 于 较 大 数字 的 前 面 ， 这 
就 是 while 循 环 继续 的 条 件 是 ahead>behind 的 原因 。 代 码 中 只 有 一 个 while 
循环 从 两 端 癌 中 间 扫 描 数 组 ， 因 此 这 种 算法 的 时 间 复 杂 度 是 O Cn) 。 


源 代码 : 
本 题 完整 的 源 代码 详 见 41 1 TwoNumbersWithSum 项 目 。 
测试 用 例 : 


e ”功能 测试 (数组 中 存在 和 为 s 的 两 个 数 ， 数 组 中 不 存在 和 为 的 
两 个 数 ) 


} 


e 特殊 输入 测试 (表示 数组 的 指针 为 NULL 指 针 ) 

看 到 应 聘 者 比较 轻松 地 解决 了 问题 还 有 时 间 剩 余 ， 有 些 面试 官 喜欢 
退 问 和 前 面 问 题 相关 但 稍微 难 一 些 的 问题 。 比 如 下 面 的 问题 就 是 一 个 例 
T: 





题目 二 : 输入 一 个 正 数 s， 打 印 出 所 有 和 为 $ 的 连续 正 数 序 
列 〈 至 少 含 有 两 个 数 ) 。 例 如 输入 15， 由 于 1 十 2 十 3 十 4 十 5 一 4 
十 5 十 6 三 7 十 8=15， 所 以 结果 打印 出 3 个 连续 序列 1 一 5、4 一 6 
和 7 一 8。 


有 了 解决 前 面 问题 的 经 验 ， 我 们 也 考虑 用 两 个 数 small 和 big 分 别 表 
示 序 列 的 最 小 值 和 最 大 值 。 首 先 把 small 初 始 化 为 1，bug 初 始 化 为 2。 如 
果 从 small 到 big 的 序列 的 和 大 于 s， 我 们 可 以 从 序列 中 去 掉 较 小 的 值 ， 也 
束 是 增 大 small 的 值 。 如 果 从 small 到 big 的 序列 的 和 小 于 s， 我 们 可 以 增 大 
big， 让 这 个 序列 包含 更 多 的 数字 。 因 为 这 个 序列 至 少 要 有 两 个 数字 ， 
我 们 一 直 增 加 small 到 (1+s) /2 为 止 。 

以 求 和 为 9 的 所 有 连续 序列 为 例 ， 我 们 先 把 small 初 始 化 为 1，big 初 
始 化 为 2。 此 时 介 于 small 和 big 之 则 的 序列 是 {1,2}， 序 列 的 和 为 3， 小 于 
9， 所 以 我 们 下 一 步 要 让 序列 包含 更 多 的 数字 。 我 们 把 big 增 加 1 变 成 3， 
此 时 序列 为 {1,2,3}。 由 于 序列 的 和 是 6， 仍 然 小 于 9， 我 们 接 下 来 再 增加 
big 变 成 4， 介 于 small 和 big 之 间 的 序列 也 随 之 变 成 {1,2,3,4}。 由 于 序列 的 
和 10 大 于 9， 我 们 要 删 去 去 序列 中 的 一 些 数字 ， 于 是 我 们 增加 small 变 成 
2， 此 时 得 到 的 序列 是 {2,3,4}， 序 列 的 和 正好 是 9。 我 们 找到 了 第 一 个 和 
为 9 的 连续 序列 ， 把 它 打 印 出 来 。 接 下 来 我 们 再 增加 big， 重 复 前 面 的 过 
程 ， 可 以 找到 第 三 个 和 为 9 的 连续 序列 {4,5}。 可 以 用 表 6.2 总 结 整 个 过 


程 。 





表 6.2 求 取 和 为 9 的 连续 序列 的 过 程 


m| an| m | e |m se| O ee 
oh bh h h w 
a ho b ha fe fe jim 
ao fho he fosfo far famm 


a b h huh Je [mmam 
s b Pe Per far [mm 

ao b ha [o far famm 
aoo d hb le b |e [mm 


E Sie eet a, RET AAAA aire 下 面 是 这 
种 思路 的 参考 代码 : 








void FindContinuousSequence (int sum) 
{ 
if(sum < 3) 
return; 


int small 
int big = 
int middle 
int curSum 


~ 


ll tl ss 


while(small < middle) 
{ 
if (curSum == sum) 
PrintContinuousSequence (small, big); 


while(curSum > sum && small < middle) 


{ 
curSum -= small; 
small ++; 
if(curSum == sum) 
PrintContinuousSequence (small, big); 
} 
big ++; 


curSum += big; 


} 


void PrintContinuousSequence (int small, int big) 
{ 
for(int i = small; i <= big; ++ i) 
DrintE ("Sai h ays 


printf ("\n"); 


在 前 面 的 代码 中 ， 求 连续 序列 的 和 应 用 了 一 个 小 技巧 。 通 常 我 们 可 
以 用 循环 求 一 个 连续 序列 的 和 ， 但 考虑 到 每 一 次 操作 之 后 的 序列 和 操作 
之 前 的 序列 相 比 大 部 分 数字 都 是 一 样 的 ， 只 是 增加 或 者 减少 了 一 个 数 
字 ， 因 此 我 们 可 以 在 前 一 个 序列 的 和 的 基础 上 求 操作 之 后 的 序列 的 和 。 
这 样 可 以 减少 很 多 不 必要 的 运算 ， 从 而 提高 代码 的 效率 。 


源 代码 : 


} 





本 题 完整 的 源 代码 详 见 41_2_ContinuesSquenceWithSum 项 目 。 
测试 用 例 : 


o 功能 测试 (存在 和 为 s 的 连续 序列 ， 如 9、100 等 ;不 存在 和 为 s 
的 连续 序列 ， 如 4、0) 。 
e 边界 值 测试 〈 连 续 序 列 的 最 小 和 3 ) 


本 题 考点 : 


o 考 奋 思考 复杂 问题 的 思维 能 力 。 应 聘 者 如 有 果 能 够 通过 一 两 个 具 
体 的 例子 找到 规律 ， 解 决 这 个 问题 融 容 易 多 了 。 

e 考 香 知识 迁移 的 能 力 。 应 聘 者 面 对 第 二 个 问题 的 时 候 ， 能 不 能 
把 解决 第 一 个 问题 的 思路 应 用 到 新 的 题目 上 ， 征 面试 官 考查 知识 迁移 能 
力 的 重要 指标 。 


面试 题 42: 翻转 单词 顺序 VS 左旋 转 字 符 串 


题目 一 :输入 一 个 英文 句子 ， 翻 苞 句 子 中 单词 的 顺序 ， 但 
单词 内 字符 的 顺序 不 变 。 为 简单 起 见 ， 标 点 符号 和 普通 字母 一 
样 处 理 。 例 如 输入 字符 串 "T am a student. "， 则 输出 "student. a 
am I". 


这 个 题目 流传 甚 广 ， 很 多 公司 都 多 次 拿 来 作 面 试题 ， 很 多 应 聘 者 也 
多 次 在 各 种 博客 或 者 书籍 上 看 到 过 通过 两 次 翻转 字符 串 的 解法 ， 于 是 很 
快 就 可 以 跟 面 试 官 解释 清楚 解 题 思 路 : 第 一 步 翻 转 句 子 中 所 有 的 字符 。 
比如 翻转 "I am a student. "中 所 有 的 字符 得 到 ".tneduts a ma I"， 此 时 不 但 
翻转 了 句子 中 单词 的 顺序 ， 连 单词 内 的 字符 顺序 也 被 翻转 了 。 第 二 步 再 
翻转 每 个 单词 中 字符 的 顺序 ， 就 得 到 了 "student. a am 1"。 这 正 是 符合 题 
目 要 求 的 输出 。 

这 种 思路 的 关键 在 于 实现 一 个 函数 以 翻转 字符 串 中 的 一 段 。 下 面 的 
国 数 Reverse 可 以 完成 这 一 功能 : 

















void Reverse(char *pBegin, char *pEnd) 
{ 
if(pBegin == NULL || pEnd == NULL) 
return; 


while (pBegin < pEnd) 

{ 
char temp = *pBegin; 
*pBegin = *pEnd; 
*pEnd = temp; 


pBegin ++, pEnd --; 
} 
} 


RERI VA SR BUC a FE BE 
词 。 这 种 思路 的 参考 代码 如 下 : 


char* ReverseSentence(char *pData) 
{ 
if (pData == NULL) 
return NULL; 


char *pBegin = pData; 


char *pEnd = pData; 

while (*pEnd != '\0') 
pEnd ++; 

pEnd--; 


// 翻转 整个 句子 
Reverse (pBegin, pEnd); 


// 翻转 句子 中 的 每 个 单词 
pBegin = pEnd = pData; 
while(*pBegin != '\0') 
{ 

if (*pBegin == ' ') 


pBegin ++; 
pEnd ++; 


} 
else if(*pEnd == ' ' || *pEnd == '\0') 


rse(pBegin, --pEnd); 


return pData; 


FRE, FRETS ot, ALBA ay DE gS 22 
来 确定 每 个 单词 的 起 始 和 终止 位 置 。 在 上 述 代 码 的 翻转 每 个 单词 阶段 ， 
指针 pBegin 指 丘 向 单词 的 第 一 个 字符 ， 而 pEnd 指 向 单词 的 最 后 一 个 字符 。 


源 代码 : 


} 


本 题 完 整 的 源 代 码 详 见 42_ 1_ReverseWordsInSentence 项 目 。 
测试 用 例 : 


e 功能 测试 〈 句 子 中 有 多 个 单词 ， 句 子 中 只 有 一 个 单词 ) 。 

e 特殊 输入 测试 〈 字 符 串 指针 为 NULL 指 针 、 字 符 串 的 内 容 为 
、 字 符 串 中 只 有 空格 ) 。 

有 经 验 的 面试 官 看 到 一 个 应 聘 者 几乎 不 假 思 索 就 能 想 出 一 种 比较 巧 
妙 的 算法 ， 就 会 觉得 他 之 前 可 能 见 过 这 个 题目 。 这 个 时 候 很 多 面试 官 都 
会 再 问 一 个 问题 ， 以 考查 他 是 不 是 真 的 理解 了 这 种 算法 。 面 试 官 一 个 常 
见 的 考查 办 法 就 是 问 一 个 类 似 的 但 更 加 难 一 点 的 问题 。 以 这 道 题 为 例 ， 
如 果 面 试 官 觉得 应 聘 者 之 前 看 过 这 个 思路 ， 那 他 可 能 再 问 第 二 个 问题 : 

题目 二 : 字符 串 的 左旋 转 操作 是 把 字符 串 前 面 的 徊 干 个 字 
符 转 移 到 字符 串 的 尾部 。 请 定义 一 个 函数 实现 字符 串 左 旋转 操 
作 的 功能 。 比 如 输入 字符 串 "abcdefg" 和 数字 2， 该 函数 将 返回 
左旋 转 2 位 得 到 的 结果 "cdefgab"。 

要 找到 字符 串 旋转 时 每 个 字符 移动 的 规律 ， 不 是 一 件 轻易 的 事情 。 
那 我 们 是 不 是 可 以 从 解决 第 一 个 问题 的 思路 中 找到 启发 ? 在 第 一 个 问题 
中 ， 如 果 输 入 的 字符 串 之 中 只 有 两 个 单词 ， 比 如 "hello world"， 那 么 翻 
转 这 个 句子 中 的 单词 顺序 就 得 到 了 "world ”hello"。 比 较 这 两 个 字符 串 ， 
我 们 是 不 是 可 以 把 "world hello" 看 成 是 把 原始 字符 串 "hello world" 的 前 面 
若干 个 字符 转移 到 后 面 ? 也 就 是 说 这 两 个 问题 是 非常 相似 的 ， 我 们 同样 
可 以 通过 翻转 字符 串 的 办 法 来 解决 第 二 个 问题 。 

以 "abcdefg" 为 例 ， 我 们 可 以 把 它 分 为 两 部 分 。 由 于 想 把 它 的 前 两 个 
字符 移 到 后 面 ， 我 们 就 把 前 两 个 字符 分 到 第 一 部 分 ， 把 后 面 的 所 有 字符 
都 分 到 第 二 部 分 。 我 们 先 分 别 翻转 这 两 部 分 ， 于 是 就 得 到 "bagfedc"。 接 
下 来 我 们 再 翻转 整个 字符 串 ， 得 到 的 "cdefgab" 刚 好 就 是 把 原始 字符 串 左 
旋转 2 位 的 结果 。 

通过 前 面 的 分 析 ， 我 们 发 现 只 需要 调用 3 次 前 面 的 Reverse 函 数 就 可 
以 实现 字符 串 的 左旋 转 功 能 。 参 考 代 码 如 下 : 


Hi 




















char* LeftRotateString(char* pStr, int n) 
{ 
if(pStr != NULL) 
{ 
int nLength = static_cast<int>(strlen(pStr)); 
if(nLength > 0 gg n > 0 && n < nLength) 


{ 


char* pFirstStart = pStr; 

chart pFirstEnd = pStr +n - T} 

char* pSecondStart = pStr + n; 

char* pSecondEnd = pStr + nLength - 1; 


// 翻转 字符 串 的 前 面 n 个 字符 

Reverse (pFirstStart, pFirstEnd); 
// 翻转 字符 串 的 后 面部 分 

Reverse (pSecondStart, pSecondEnd); 
// PAREN HF 

Reverse (pFirstStart, pSecondEnd); 


想 清楚 思路 之 后 再 写 代 码 是 一 件 很 容易 的 事情 ， 但 我 们 也 不 能 掉 以 
轻 心 。 面 试 官 在 检查 与 字符 串 相 关 的 代码 时 经 常会 发 现 两 种 问题 ， 一 是 
输入 空 指 针 NULL 时 程序 会 骨 演 ， 二 是 内 存 访问 越界 的 问题 ， 也 就 是 试 
图 访问 不 属于 字符 串 的 内 存 。 例 如 如 果 输 入 的 n 小 于 0， 指 针 pStr+n 指 癌 
的 内 存 就 不 属于 字符 串 。 如 果 我 们 不 排除 这 种 情况 ， 试 图 访问 不 属于 字 
符 串 的 内 存 就 会 留 下 严重 的 内 存 越界 的 安全 隐患 。 在 前 面 的 代码 中 ， 我 
们 添加 了 两 个 和 f 判 断 语句 就 是 为 了 防止 出 现 这 两 种 问题 。 


源 代码 : 
本 题 完 整 的 源 代码 详 见 42_2_LeftRotateString 项 目 。 

















MAH P: 
。 功能 测试 (把 长 度 为 p 的 字符 串 左 旋转 0 个 字符 、1 个 字符 、2 个 


= AY 


字符 、n 一 1 个 字符 、n 个 字符 、n 十 1 个 字符 ) 。 
。 特殊 输入 测试 字符 串 的 指针 为 NULL 指 针 ) 。 


本 题 考点 : 


e 考 碍 知识 迁移 的 能 力 。 当 面试 的 时 候 遇 到 第 二 个 问题 ， 而 之 前 
我 们 做 过 “翻转 句子 中 单词 的 顺序 ”这 个 题目 ， 那 如 果 能 够 把 多 次 翻转 字 
符 串 的 思路 迁移 过 来 ， 就 能 很 轻易 地 解决 字符 串 左 旋转 的 问题 。 

© 考 奋 对 字符 串 的 编程 能 力 。 


6.4 抽象 建 模 能 


计算 机 只 是 一 种 工具 ， 它 的 作用 是 用 来 解决 实际 生产 生活 中 的 问 
题 。 程 序 员 的 工作 就 是 把 各 种 现实 问题 抽象 成 数学 模型 并 用 计算 机 的 编 
程 语 言 表达 出 来 ， 因 此 有 些 面试 官 喜 欢 从 日 常生 活 中 抽取 提炼 出 问题 考 
查 应 聘 者 是 否 能 建立 数学 模型 并 解决 问题 。 要 想 顺 利 解决 这 种 类 型 的 问 
题 ， 应 聘 者 除了 需要 具备 扎实 的 数学 基础 和 编程 能 力 之 外 ， 还 需要 具有 
敏锐 的 洞察 力 和 丰富 的 想象 力 。 

建 模 的 第 一 步 是 选择 合理 的 数据 结构 来 表述 问题 。 实 际 生产 生活 中 
的 问题 干 变 万 化 ， 而 常用 的 数据 结构 却 只 有 有 限 的 几 种 。 我 们 在 根据 问 
题 的 特点 综合 考虑 性 能 、 编 程 难度 等 因素 之 后 ， 选 择 最 合适 的 数据 结构 
来 表达 问题 ， 也 就 是 建立 模型 。 比 如 在 面试 题 44“ 扑 克 牌 的 顺 子 ” 中 ， 我 
们 用 一 个 数组 表示 一 副 牌 ， 用 11、12 和 13 分 别 表示 J、Q、K 并 且 用 0 表 
示 大 小 王 。 在 面试 题 45“ 圆 峰 中 最 后 剩 下 的 数字 ”中 ， 我 们 可 以 用 一 个 环 
形 链表 模拟 一 个 圆圈 。 

建 模 的 第 二 步 是 分 析 模 型 中 的 内 在 规律 ， 并 用 编程 语言 表述 这 种 规 
律 。 我 们 只 有 对 现实 问题 进行 深入 细微 的 观察 分 析 之 后 ， 才 能 找到 模型 
中 的 规律 ， 才 有 可 能 编程 解决 问题 。 例 如 在 本 书 2.4.2 节 提 到 的 “青蛙 跳 
台阶 ”问题 中 ， 它 内 在 的 规律 是 斐 波 那 契 数列 。 再 比如 面试 题 43“n 个 般 
子 的 点 数 ” 问 题 ， 其 本 质 是 求 数列 f Cn) =f (Cn 一 1) +f Cn 一 2) +f Cn 
一 3) +f Cn 一 4) +f Cn 一 5) +f Cn 一 6) 。 找 到 这 个 规律 之 后 ， 我 们 
就 可 以 分 别 用 递归 和 循环 两 种 不 同 的 方法 去 写 代 码 。 然 而， 并 不 是 所 有 
问题 的 内 在 规律 都 是 显而易见 的 。 在 面试 题 45“ 圆 圈 中 最 后 剩 下 的 数 
字 ” 中 ， 我 们 经 过 严密 的 数学 分 析 之 后 才能 找到 每 次 从 圆圈 中 删除 的 数 
字 的 规律 ， 从 而 找到 一 种 不 需要 辅助 环形 链表 的 快速 方法 来 解决 问题 。 


面试 题 43: nh RTH eA 


题目 ， 把 n 个 人 般 子 扔 在 地 上 ， 所 有 仍 子 旨 上 一 面 的 点 数 之 
和 为 s。 输 入 n， 打 印 出 s 的 所 有 可 能 的 值 出 现 的 概率 。 
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大 值 为 6n。 另 外 根据 排列 组 合 的 知识 ， 我 们 还 知道 n 个 般 子 的 所 有 点 数 
的 排列 数 为 66。 要 解决 这 个 问题 ， 我 们 需要 先 统计 出 每 一 个 点 数 出 现 的 
0 68 5 
LS 。 


解法 一 : 基于 递归 求 般 子 点 数 ， 时 间 效 率 不 够 高 


现在 我 们 考虑 如 何 统计 每 一 个 点 数 出 现 的 次 数 。 要 想 求 出 n 个 角子 
的 点 数 和 ， 可 以 先 把 n 个 骨 子 分 为 两 堆 : 第 一 堆 只 有 一 个 ， 男 一 个 有 n 一 
1 个 。 单 独 的 那 一 个 有 可 能 出 现 从 1 到 6 的 点 数 。 我 们 需要 计算 从 1 到 6 的 
每 一 种 点 数 和 剩 下 的 n 一 1 个 人 般 子 来 计算 点 数 和 。 接 下 来 把 剩 下 的 n 一 1 个 
角子 还 是 分 成 两 堆 ， 第 一 堆 只 有 一 个 ， 第 二 堆 有 n 一 2 个 。 我 们 把 上 一 轮 
那个 单独 骨 子 的 点 数 和 这 一 轮 单 独 角 子 的 点 数 相 加 ， 再 和 剩 下 的 n 一 2 个 
骨 子 来 计算 点 数 和 。 分 析 到 这 里 ， 我 们 不 难 发 现 这 是 一 种 递归 的 思路 ， 
加 归结 束 的 条 件 束 是 最 后 只 剩 下 一 个 般 子 。 

我 们 可 以 定义 一 个 长 度 为 6n 一 n 十 1 的 数组 ， 和 为 的 点 数 出 现 的 次 
数 保存 到 数组 第 s 一 n 个 元 素 里 。 基 于 这 种 思路 ， 我 们 可 以 写 出 如 下 代 
AS: 











int g maxValue = 6; 
void PrintProbability(int number) 
{ 
if(number < 1) 
return; 


int maxSum = number * g_maxValue; 

int* pProbabilities new int[maxSum - number + 1]; 

for(int i = number; i <= maxSum; ++i) 
pProbabilities[i - number] = 0; 


Probability(number, pProbabilities); 


int total = pow((double)g_maxValue, number); 

for(int i = number; i <= maxSum; ++i) 

{ 
double ratio = (double) pProbabilities[i - number] / total; 
printf("%td: %e\n", i, ratio); 


} 


delete[] pProbabilities; 
} 


void Probability(int number, int* pProbabilities) 
{ 
for(int i = 1; i <= g_maxValue; ++i) 
Probability(number, number, i, pProbabilities); 
} 


void Probability(int original, int current, int sum, 
int* pProbabilities) 
{ 
if(current == 1) 
{ 
pProbabilities[sum - original]+t+; 
} 
else 
{ 
for(int i= 1; i <= g_maxValue; ++i) 
{ 
Probability (original, current -1, i+ sum, pProbabilities); 
} 








上 述 思 路 很 简洁 ， 实 现 起 来 也 容易 。 但 由 于 是 基于 递归 的 实现 ， 它 
有 很 多 计算 是 重复 的 ， 从 而 导致 当 number 变 大 时 性 能 慢 得 让 人 不 能 接 
受 。 关 于 递归 的 性 能 讨论 ， 详 见 本 书 2.4.2 市 。 


解法 二 : 基于 循环 求 角 子 点 数 ， 时 间 性 能 好 


可 以 换 一 种 思路 来 解决 这 个 问题 。 我 们 可 以 考虑 用 两 个 数组 来 存储 
仍 子 点 数 的 每 一 个 总 数 出 现 的 次 数 。 在 一 次 循环 中 ， 第 一 个 数组 中 的 第 
n 个 数字 表示 骨 子 和 为 n 出 现 的 次 数 。 在 下 一 循环 中 ， 我 们 加 上 一 个 新 的 
人 般 子 ， 此 时 和 为 n 的 人 般 子 出 现 的 次 数 应 该 等 于 上 一 次 循环 中 般 子 点 数 和 
为 n 一 1、n 一 2、n 一 3、n 一 4、n 一 5 与 n 一 6 的 次 数 的 总 和 ， 所 以 我 们 把 另 
一 个 数组 的 第 n 个 数字 设 为 前 一 个 数组 对 应 的 第 n 一 1、n 一 2、n 一 3、n 一 
4、n 一 5 与 n 一 6 之 和 。 基 于 这 个 思路 ， 我 们 可 以 写 出 如 下 代码 : 





void PrintProbability(int number) 
{ 
if (number < 1) 
return; 


int* pProbabilities[2]; 

pProbabilities[0] = new int[g_maxValue * number + 1]; 
pProbabilities[1] = new int[g_maxValue * number + 1]; 
for(int i = 0; i < g_maxValue * number + 1; ++i) 

{ 

0; 

0; 


pProbabilities[0] [i] 
pProbabilities[1] [i] 


} 


int flag = 0; 
for (int i = 1; i <= g_maxValue; ++i) 
pProbabilities[flag] [i] = 1; 


for (int k = 2; k <= number; ++k) 
{ 
for(int i = 0; 1 < kr ++i) 
pProbabilities[1l - flag] [i] = 0; 


for (int i = k; i <= g_maxValue * k; ++i) 
{ 
pProbabilities[1l - flag][i] = 0; 
for(int j = 1; j <= i && j <= g_maxValue; ++]) 
pProbabilities[1-flag] [i]+=pProbabilities[flag] [i-j]; 
} 


flag = 1 - flag; 
} 


double total = pow((double)g_maxValue, number); 

for(int i = number; i <= g_maxValue * number; ++i) 

{ 
double ratio = (double) pProbabilities[flag] [i] / total; 
printf("td: te\n", i, ratio); 


} 


delete[] pProbabilities([0]; 
delete[] pProbabilities([1]; 


在 上 述 代码 中 ， 我 们 定义 了 两 个 数组 pProbabilities[0] 和 
pProbabilities[1] 来 存储 货 子 的 点 数 之 和 。 在 一 轮 循 环 中 ， 一 个 数组 的 第 
n 项 等 于 另 一 数组 的 第 n 一 1、n 一 2、n 一 3、n 一 4、n 一 5 以 及 n 一 6 项 的 
和 。 在 下 一 轮 循环 中 ， 我 们 交换 这 两 个 数组 (通过 改变 变量 flag 实 现 ) 
再 重复 这 一 计算 过 程 。 

值得 注意 的 是 ， 上 述 代码 没有 在 函数 里 把 一 个 般 子 的 最 大 点 数 硬 编 
人 码 Chard code) 为 6， 而 是 用 一 个 变量 g_maxValue 来 表示 。 这 样 做 的 好 
处 是 ， 如 果 某 个 三 家 生产 了 其 他 点 数 的 最 子 ， 我 们 只 需要 在 代码 中 修改 
一 个 地 方 ， 扩 展 起 来 很 方便 。 如 果 在 面试 的 时 候 我 们 能 对 面试 官 提 起 对 
程序 扩展 性 的 考虑 ， 一 定 能 给 面试 官 留 下 很 好 的 印象 。 

















源 代码 : 
本 题 完整 的 源 代码 详 见 43_DicesProbability 项 目 。 
测试 用 例 : 


e 功能 测试 1、2、3、4 个 人 般 子 的 各 点 数 的 概率 ) 。 
。 特殊 输入 测试 〈 输 入 0) 。 
。 性 能 测试 “输入 较 大 的 数字 ， 比 如 11) 。 


本 题 考点 : 


e 数学 建 模 的 能 力 。 不 管 采 用 哪 种 思路 解决 问题 ， 我 们 都 要 先 想 
到 用 数组 来 存放 n 个 般 子 的 每 一 个 扣 数 出 现 的 次 数 ， 并 通过 分 析 点 数 的 
规律 建 并 模型 并 最 终 找 到 解决 方案 。 

© 考查 对 递归 和 循环 的 性 能 的 理解 。 


面试 题 44: fh oe REY 


题目 ;从 扑克 牌 中 随机 抽 5 张 牌 ， 判 断 是 不 是 一 个 顺 子 ， 
即 这 5 张 牌 是 不 是 连续 的 。2 一 10 为 数字 本 身 ，A 为 1，J 为 11， 
Q 为 12，K 为 13， 而 大 、 小 王 可 以 看 成 任意 数字 。 

我 们 需要 把 扑 殉 牌 的 背景 抽象 成 计算 机 语言 。 不 难 想 象 ， 我 们 可 以 
把 5 张 牌 看 成 由 5 个 数字 组 成 的 数组 。 大 、 小 王 是 特殊 的 数字 ， 我 们 不 妨 
把 它们 都 定义 为 0， 这 样 就 能 和 其 他 扑 殉 牌 区 分 开 来 了 。 

接 下 来 我 们 分 析 怎 样 判断 5 个 数字 是 不 是 连续 的 ， 最 直观 的 方法 是 
把 数组 排序 。 值 得 注意 的 是 ， 由 于 0 可 以 当成 任意 数字 ， 我 们 可 以 用 0 去 
补 满 数组 中 的 空缺 。 如 果 排 序 之 后 的 数组 不 是 连续 的 ， 即 相 邻 的 两 个 数 




















字 相 隔 若 干 个 数字 ， 但 只 要 我 们 有 足够 的 0 可 以 补 满 这 两 个 数字 的 空 
缺 ， 这 个 数组 实际 上 还 是 连续 的 。 举 个 例子 ， 数 组 排序 之 后 为 {0，1， 
3，4，5}， 在 1 和 3 之 间 空 缺 了 一 个 2， 刚 好 我 们 有 一 个 0， 也 就 是 我 们 可 
以 把 它 当 成 2 去 填补 这 个 空缺 。 

于 是 我 们 需要 做 3 件 事 情 : 首先 把 数组 排序 ， 再 统计 数组 中 0 的 个 
数 ， 最 后 统计 排序 之 后 的 数组 中 相 邻 数字 之 间 的 空缺 总 数 。 如 果 空 缺 的 
eae et Onna 那么 这 个 数组 就 是 连续 的 ， 反 之 则 不 连 
续 。 

最 后 ， 我 们 还 需要 注意 一 点 : 如 果 数 组 中 的 非 0 数 字 重 复出 现 ， 则 
该 数组 不 是 连续 的 。 换 成 扑克 牌 的 描述 方式 就 是 如 果 一 副 牌 里 含有 对 
子 ， 则 不 可 能 是 顺 子 。 

基于 这 个 思路 ， 我 们 可 以 写 出 如 下 代码 : 

















bool IsContinuous(int* numbers, int length) 
{ 
1f(numbers == NULL || length < 1) 
return false; 


qsort (numbers, length, sizeof(int), compare); 


int numberOfZero = 0; 
int numberOfGap = 0; 


// 统计 数组 中 0 的 个 数 
for(int i = 0; i < length && numbers[i] == 0; ++i) 
++ numberOfZero; 


// Airin p 64 i hak E 
int small = numberOfZero; 
int big = small + 1; 
while (big < length) 
{ 
// 两 个 数 相 等 ， 有 对 子 ， 不 可 能 是 顺 子 
if (numbers[small] == numbers [big]) 


return false; 


numberOfGap += numbers[big] - numbers [small] - 1; 
small = big; 
++big; 


} 


return (numberOfGap > numberOfZero) ? false : true; 


} 


int compare(const void *argl, const void *arg2) 
{ 


return *(int*)argl - *(int*)arg2; 


为 了 让 代码 显得 简洁 ， 上 述 代 码 调 用 C 的 库 函 数 qsort 排 序 。 可 能 
人 担心 qsort 是 O Cnlogn) 的 时 间 复 杂 度 ， 还 不 够 快 。 由 于 扑 殉 牌 的 值 出 
现在 0 一 13 之 间 ， 我 们 可 以 定义 一 个 长 度 为 14 的 哈 希 表 ， 这 样 在 O (n) 
时 间 惑 能 完成 排序 〈 本 书 2.4.1 节 有 这 种 思路 的 例子 ) 。 通 名 我 们 认为 不 
同 级 别 的 时 间 复 杂 度 只 有 当 n 足 够 大 的 时 候 才 有 意义 。 由 于 本 题 中 数组 
的 长 度 是 固定 的 ， 只 有 5 张 牌 ， 那 么 O a) MO alog) 不 会 有 多 少 区 














别 ， 我 们 可 以 选用 简洁 易 慌 的 方法 来 实现 算法 。 


源 代码 : 
本 题 完整 的 源 代码 详 见 44_ContinousCards 项 目 。 
测试 用 例 : 


e 功能 测试 〈 抽 出 的 牌 中 有 一 个 或 者 多 个 大 、 小 王 ， 抽 出 的 牌 中 
没有 大 、 小 王 ， 抽 出 的 牌 中 有 对 子 ) 。 

e 特殊 输入 测试 (输入 NULL 指 针 )。 

本 题 考点 : 

考查 抽象 建 模 能 力 。 这 个 题 日 要 求 我 们 把 熟悉 的 扑克 有 牌 转换 为 数 
组 ， 把 找 顺 子 的 过 程 通过 排序 、 计 数 等 步骤 实现 。 这 些 都 是 把 生活 中 的 
模型 用 程序 语言 来 表达 的 例子 。 


面试 题 45: 中 最 后 剩 下 的 数字 


题目 : 0,1...n 一 1 这 n 个 数字 排 成 一 个 圆圈 ， 从 数字 0 开始 
每 次 从 这 个 圆圈 里 删除 第 m 个 数字 。 求 出 这 个 圆圈 里 剩 下 的 最 
后 一 个 数字 。 

例如 ，0、1、2、3、4 这 5 个 数字 组 成 一 个 圆圈 (如 图 6.2 所 示 )， 
从 数字 0 开始 每 次 删除 第 3 个 数字 ， 则 删除 的 前 四 个 数字 依次 是 2、0、 
4、1， 因 此 最 后 剩 下 的 数字 是 3。 











图 6.2 ”由 0 一 4 这 5 个 数字 组 成 的 圆圈 
本 题 就 是 有 名 的 约瑟夫 (Josephuse) 环 问题 。 我 们 介绍 两 种 方法 : 
一 种 方法 是 用 环形 链表 模拟 的 经 典 解法 ， 第 二 种 方法 是 分 析 每 次 被 
删除 的 数字 的 规律 并 直接 计算 出 中 最 后 剩 下 的 数字 。 














经 典 的 解法 ， 用 环形 链表 模拟 


既然 题目 中 有 一 个 数字 圆圈 ， 很 自然 的 想法 就 是 用 一 个 数据 结构 来 
模拟 这 个 圆圈 。 在 常用 的 数据 结构 中 ， 我 们 很 容易 想到 环形 链表 。 我 们 
ee 然后 每 次 在 这 个 链表 中 删除 
mMm SJ ZA Fi o 

如 果 面 试 官 要 求 我 们 不 能 使 用 标准 模板 库 里 的 数据 容器 来 模拟 环形 
链表 ， 我 们 自己 实现 一 个 链表 也 不 是 很 难 的 事情 。 如 果 面 试 官 没 有 特殊 
要 求 ， 我 们 就 可 以 用 模板 库 中 的 std::list 来 模拟 一 个 环形 链表 。 由 于 
std::list 本 丑 并 不 是 一 个 环形 结构 ， 因 此 每 当 和 迭代 器 (Iterator〉 扫 接 到 链 
表 末 尾 的 时 候 ， 我 们 要 记得 把 迭代 器 移 到 链表 的 涉 部 ， 这 样 就 相当 于 按 
照 顺 序 在 一 个 圆圈 里 裔 历 了 。 这 种 思路 的 代码 如 下 : 





int LastRemaining(unsigned int n, unsigned int m) 
{ 
Lis 2 a ee 1 
return 一 二 ;7 
unsigned int i = 0; 
list<int> numbers; 
for(i = 0; & wmr ++ 1) 
numbers.push_back(i); 


List<int>::it 
while (numbers.s 


current ++; 
if(current == numbers.end()) 
current = numbers.begin(); 


ist<int>::iterator next = ++ current; 
next == numbers.end()) 
next = numbers.begin(); 


zm current; 
numbers .erase (current) 
current = next; 


} 


return * (current); 


} 





如 果 我 们 用 一 两 个 例子 仔细 分 析 上 述 代码 的 运行 过 程 ， 束 会 发 现 我 
们 实际 上 需要 在 环形 链表 里 重复 志 历 很 多 这。 重复 的 吉 历 当然 对 时 间 效 
率 有 人 负面 的 影响 。 这 种 方法 每 删除 一 个 数字 需要 m 步 运算 ， 忌 共有 n 个 
数字 ， 因 此 总 的 时 间 复 杂 度 是 O (mn) 。 同 时 这 种 思路 还 需要 一 个 辅助 
链表 来 模拟 圆圈 ， 其 空间 复杂 度 是 O(n) 。 接 下 来 我 们 试看 找到 每 次 
被 删除 的 数字 有 哪些 规律 ， 和 希望 能 够 找到 更 加 高 效 的 算法 。 


创新 的 解法 ， 拿 到 Offer 不 在 话 下 
首先 我 们 定义 一 个 关于 n 和 m 的 方程 f (n,m) ， 表 示 每 次 在 n 个 数字 











0,1...n 一 1 中 每 次 删除 第 m 个 数字 最 后 剩 下 的 数字 。 

在 这 n 个 数字 中 ， 第 一 个 被 删除 的 数字 是 (m1) %n。 为 了 简单 起 
见 ， 我 们 把 (m—1) %n 记 为 k， 那 么 删除 k 之 后 剩 下 的 n 一 1 个 数字 为 0,1， 
.水 一 1 十 1...n 一 1， 并 且 下 一 次 删除 从 数字 k 十 1 开始 计数 。 相 当 于 在 
剩 下 的 序列 中 ，k 二 1 排 在 最 前 面 ， 从 而 形成 k 十 1,...,n 一 1,0,1,...,k 一 1。 
该 序列 最 后 剩 下 的 数字 也 应 该 是 关于 n 和 mm 的 函数 。 由 于 这 个 序列 的 规 
律 和 前 面 最 初 的 序列 不 一 样 〈 最 初 的 序列 是 从 0 开始 的 连续 序列 ) ， 
此 该 函数 不 同 于 前 面 的 函数 ， 记 为 f (n 一 1,m) 。 最 初 序列 最 后 剩 下 的 
数字 f (n,m) 一 定 是 删除 一 个 数字 之 后 的 序列 最 后 剩 下 的 数字 ， 即 
f (nm) =f? (n—-1,m) 。 

fee BOREAL FE Fn 1 SAE A Pk -+-1,....n-1,0,1,...,.k—-1 
做 一 个 映射 ， 映 射 的 结果 是 形成 一 个 从 0 到 n 一 2 的 序列 : 


k+1 > 0 














k+2 > 1 


k~1 > n-2 

我 们 把 映射 定义 为 p， 则 p (x) = (x—-k—-1) %n。 它 表示 如 果 映 
射 前 的 数字 是 x， 那 么 映射 后 的 数字 是 〈Xx 一 k 一 1) %n。 该 映射 的 逆 映 
Keep (x) = (x 十 k 十 1) %n. 

由 于 映射 之 后 的 序列 和 最 初 的 序列 共有 同样 的 形式 ， 即 都 是 从 0 开 
始 的 连续 序列 ， 因 此 仍然 可 以 用 函数 f 来 表示 ， 记 为 f Cn 一 Lm) 。 根 据 
我 们 的 映射 规则 ， 英 射 之 前 的 序列 中 最 后 剩 下 的 数字 f (n 一 1,m) =p 
Tf Cn 一 Lm) J=[f Cn 一 1Lm) 十 k 十 1]%n， 把 k= (m—1) %n 代 入 得 到 
f (nm) =f? (n—1,m) =[f (n—-1,m) +m]%n. 

经 过 上 面 复 杂 的 分 析 ， 我 们 终于 找到 了 一 个 递归 公式 。 要 得 到 n 个 
数字 的 序列 中 最 后 剩 下 的 数字 ， 只 需要 得 到 n 一 1 个 数字 的 序列 中 最 后 剩 
下 的 数字 ， 并 以 此 类 推 。 当 n= 二 1 时 ， 也 就 是 序列 中 开始 只 有 一 个 数字 
0， 那 么 很 显然 最 后 剩 下 的 数字 就 是 0。 我 们 把 这 种 关系 表示 为 : 




















n=l 





40 
f (n,m) 7 Ha seach n>l 
这 个 公式 无 论 用 递归 还 是 用 循环 ， 都 很 容易 实现 。 下 面 是 一 段 基 于 
循环 实现 的 代码 : 
int LastRemaining(unsigned int n, unsigned int m) 
{ 
EIn | me 2) 
return —1; 
int last = 0; 
for (int i = 2z 1 <<: t ++) 
last = (last + m) p 
return last; 








可 以 看 出 ， 这 种 思路 的 分 析 过 程 尽 管 非常 复杂 ， 但 写 出 的 代码 却 非 
第 简洁 ， 这 就 是 数学 的 魅力 。 最 重要 的 是 ， 这 种 算法 的 时 间 复 杂 度 是 
O(n) ， 空 间 复 杂 上 度 是 O (1) ， 因 此 无 论 在 时 间 效 率 还 是 空间 效率 上 


都 优 于 第 一 种 方法 。 











源 代码 : 
本 题 完整 的 源 代码 详 见 45_LastrNumberInCircle 项 目 。 
测试 用 例 : 


e 功能 测试 〈 输 入 的 mm 小 于 n， 比 如 从 最 初 有 5 个 数字 的 圆圈 删除 
每 次 第 >、3 个 数字 ; 输入 的 m 大 于 或 者 等 于 n， 比 如 从 最 初 有 6 个 数字 的 


圆圈 删除 每 次 第 6、7 个 数字 ) 。 
RATA CER OTE) 。 





@ 
e 性 能 测试 〈 从 最 初 有 4000 个 数字 的 圆圈 中 每 次 删除 第 997 个 数 
本 题 考点 : 


o 考 俘 抽 象 建 模 的 能 力 。 不 管 应 聘 者 是 用 环形 链表 来 模拟 圆圈， 
还 是 分 析 被 删除 数字 的 规律 ， 部 要 深刻 理解 这 个 问题 的 特点 并 编程 实现 


目 己 的 解决 方案 。 
© 考 奋 对 环形 链表 的 理解 及 应 用 能 力 。 大 部 分 面试 官 只 要 求 应 聘 
者 基于 环形 链表 的 方法 解决 这 个 问题 。 

考查 数学 功底 及 逻辑 思维 能 力 。 少 数 对 算法 和 数学 基础 要 求 很 


高 的 公司 ， 面 试 官 会 要 求 应 聘 者 不 能 使 用 O Cn) 的 辅助 内 存 ， 这 个 时 
候 应 聘 痢 整 只 能 静 下 心 来 一 步 步 推 寻 出 每 次 删除 的 数字 有 哪些 规律 。 


6.5 ”发散 思维 能 


发 散 思 维 的 特点 是 思维 活动 的 多 向 性 和 变通 性 ， 也 就 是 我 们 在 思考 
问题 时 注重 运用 多 思路 、 多 方案 、 多 途径 地 解决 问题 。 对 于 同一 个 问 
题 ， 我 们 可 以 从 不 同 的 方向 、 侧 面 和 层次 ， 采 用 探索 、 转 换 、 迁 移 、 组 
合 和 分 解 等 方法 ， 提 出 多 种 创新 的 解法 。 

通过 考查 友 散 思维 能 力 ， 面 试 官能 够 了 解 应 聘 者 探索 新 思路 的 激 
情 。 面 试 时 面试 官 故意 限制 应 聘 者 不 能 使 用 常规 的 思路 ， 此 时 他 在 观察 
应 聘 者 有 没有 积极 的 心态 ， 是 不 是 能 够 主动 跳出 常规 思维 的 束缚 从 多 角 
度 去 思考 问题 。 比 如 在 面试 题 46“ 求 1 十 2 十 ... 十 n” 中 ， 面 试 官 有 意 限制 
不 能 使 用 乘除 法 及 与 循环 、 条 件 判 断 、 选 择 相 关 的 关键 字 。 这 个 问题 应 
该 说 是 很 难 的 。 在 难题 面前 ， 应 聘 者 是 轻 言 放弃 ， 还 是 充满 激情 地 寻找 
a 
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常规 思路 过 到 阻碍 的 时 候 ， 应 聘 者 能 不 能 及 时 地 从 为 外 的 角度 用 不 同 的 
方法 去 分 析 问 题 ， 这 些 都 能 体现 应 聘 者 的 创造 力 。 在 面试 题 47“ 不 用 加 
减 乘 除 做 加 法 ?中 ， 当 四 则 运算 被 限制 使 用 的 时 候 ， 应 聘 者 能 不 能 迅速 
人 
现 。 

通过 考查 发 散 思 维 ， 面 试 官 还 能 了 解 应 聘 者 知识 面 的 广度 和 深度 。 
面试 实际 上 是 一 个 厚积薄发 的 过 程 。 在 遇 到 问题 之 后 ， 应 聘 者 如 果 有 共有 
览 泛 的 知识 面 并 且 对 各 领域 有 较 深 的 理解 ， 那 么 他 就 更 容易 从 不 同 的 角 
度 去 思考 问题 。 比 如 我 们 可 以 从 构造 函数 、 虚 函数 、 函 数 指针 及 模板 参 
数 的 实例 化 等 不 同 角度 去 解决 面试 题 46“ 求 1 十 2 十 … 十 nm"。 只 有 对 C++ 各 
方面 的 特性 了 如 指 掌 ， 我 们 才能 在 遇 到 问题 的 时 候 将 各 个 知识 点 信 手 折 
来 。 同 样 ， 如 宁 我 们 在 学 习 数 字 电 路 相关 谍 程 的 时 候 对 CPU 中 加 法 器 的 
原理 有 深刻 的 理解 ， 那 么 自然 就 会 想到 从 二 进 制 和 位 运算 的 角度 去 思考 
解决 面试 题 47“ 不 用 加 减 乘除 做 加 法 ”。 


面试 题 46: 求 1 十 2 十 ... 十 n 
题目 : 求 1 十 2 十 ... 十 n， 要 求 不 能 使 用 乘除 法 、for、 






































while、if、else、switch、case 等 关键 字 及 条 件 判 断 语 句 CA? 
BGs 

这 个 问题 本 身 没有 太 多 的 实际 意义 ， 因 为 在 软件 开发 中 不 可 能 有 这 
么 苛刻 的 限制 。 但 不 少 面 试 官 认 为 这 是 一 道 不 错 的 能 够 考查 应 聘 者 发 散 
思维 能 力 的 题目 ， 而 发 散 思 维 能 够 反映 出 应 聘 者 知识 面 的 宽度 ， 以 及 对 
编程 相关 技术 理解 的 深度 。 

通常 求 1 十 2 十 ... 十 n 除 了 用 公式 n Cn 十 1) /2 之 外 ， 无 外 平 循环 和 递 
归 两 种 思路 。 由 于 已 经 明确 限制 for 和 while 的 使 用 ， 循 环 已 经 不 能 再 用 
了 。 递 归 函 数 也 需要 用 证 语句 或 者 条 件 判 断 语 句 来 判断 是 继续 递归 下 去 
还 是 终止 递归 ， 但 现在 题目 已 经 不 允许 使 用 这 两 种 语句 了 。 


解法 一 : 利用 构造 浮 数 求 解 


我 们 仍然 围绕 循环 做 文章 。 循 环 只 是 让 相同 的 代码 重复 执行 n 志 而 
己 ， 我 们 完全 可 以 不 用 for 和 while 来 达到 这 个 效果 。 比 如 我 们 先 定义 一 
个 类 型 ， 接 着 创建 n 个 该 类 型 的 实例 ， 那 么 这 个 类 型 的 构造 函数 将 确定 
会 被 调用 n 次 。 我 们 可 以 将 与 罕 加 相关 的 代码 放 到 构造 函数 里 。 如 下 代 
码 正 是 基于 这 个 思路 : 














class Temp 
{ 
public: 
Temp() { ++ N; Sum += N; } 


static void Reset() { N = 0; 
static unsigned int GetSum( 


private: 
static unsigned int N; 
Static unsigned int Sum; 
}; 


unsigned int Temp::N = 0; 
unsigned int Temp::Sum = 0; 


> Sum = O's 
i() { return Sum; 


} 


unsigned int Sum_Solutionl(unsigned int n) 


{ 
Temp: :Reset (); 


Temp *a = new Temp[n]; 
delete []a; 
a = NULL; 


解法 二 : 利用 虚 函 数 求解 


我 们 同样 也 可 以 围绕 递归 做 文章 。 既 然 不 能 在 一 个 函数 中 判断 是 不 
是 应 该 终止 递归 ， 那 么 我 们 不 妨 定义 两 个 水 数 ， 一 个 函数 充当 递归 函数 
的 角色 ， 男 一 个 函数 处 理 终止 递归 的 情况 ， 我 们 需要 做 的 就 是 在 两 个 函 
数 里 二 选 一 。 从 二 选 一 我 们 很 自然 地 想到 布尔 变量 ， 比 如 值 为 ture (1) 
的 时 候 调用 第 一 个 函数 ， 值 为 false (0〉 的 时 候 调 用 第 二 个 函数 。 那 现 
在 的 问题 是 如 何 把 数值 变量 n 转 换 成 布尔 值 。 如 果 对 n 连 续 做 两 次 反 运 
算 ， 即 !m， 那 么 非 零 的 n 转 换 为 tue，0 转 换 为 false。 有 了 上 述 分 析 ， 我 


们 再 来 看 下 面 的 代码 : 


} 


class A; 
A* Array([2]; 


class A 
{ 
public: 
virtual unsigned int Sum (unsigned int n) 
{ 
return 0; 
} 
}; 


class B: public A 
{ 
public: 
virtual unsigned int Sum (unsigned int n) 
{ 
return Array[!!n]->Sum(n-1) + n; 
} 
}; 


int Sum_Solution2 (int n) 


{ 


A a; 

B b; 

Array[0] = &a; 

Array[1] = &b; 

int value = Array[1]->Sum(n); 


return value; 


这 种 思路 是 用 虚 函 数 来 实现 函数 的 选择 。 当 n 不 为 零 时 ， 调 用 水 数 
B::Sum; “né Fon, WH KAA: Sum. 


解法 三 :利用 函数 指针 求解 


在 纯 C 语 言 的 编程 环境 中 ， 我 们 不 能 使 用 虚 函 数 ， 此 时 可 以 用 函数 
指针 来 模拟 ， 这 样 代码 可 能 还 更 加 直观 一 些 : 


} 








typedef unsigned int (*fun) (unsigned int); 


unsigned int Solution3 Teminator (unsigned int n) 
{ 


return 0; 


} 

unsigned int Sum Solution3(unsigned int n) 

{ 
static fun f[2] = {Solution3 Teminator, Sum _Solution3} 
return n + f£[{!!n]j(n = 1); 

} 





解法 四 : 利用 模板 类 型 求解 


另外 我 们 还 可 以 让 编译 器 帮助 完成 类 似 于 递归 的 计算 。 比 如 如 下 代 
个 : 
template <unsigned int n> struct Sum_Solution4 


{ 
enum Value { N = Sum_Solution4<n —- 1>::N + n}; 


}; 


template <> struct Sum Solution4<1> 
{ 

enum Value { N = 1}; 
} ; 

Sum_Solution4<100>::N 就 是 1 十 2 十 ... 十 100 的 结果 。 当 编译 器 看 到 
Sum_Solution4<100> 时 ， 就 会 为 模板 类 Sum _Solution4 以 参数 100 生 成 该 
类 型 的 代码 。 但 以 100 为 参数 的 类 型 需要 得 到 以 99 为 参数 的 类 型 ， 因 为 
Sum_Solution4<100>::N= Sum _Solution4 <99>::N 十 100。 这 个 过 程 会 递 
归 一 直到 参数 为 1 的 类 型 ， 由 于 该 类 型 已 经 显 式 定义 ， 编 详 右 无 须 生 
成 ， 递 归 编 详 到 此 结束 。 由 于 这 个 过 程 是 在 编译 过 程 中 完成 的 ， 因此 要 
求 输 入 n 必 须 是 在 编译 期 间 就 能 确定 的 笛 量 ， 不 能 动态 输入 ， 这 是 该 方 
法 最 大 的 缺点 。 而 且 编译 器 对 递归 编译 代码 的 递归 深度 是 有 限制 的 ， 也 
就 是 要 求 n 不 能 太 大 。 

源 代码 : 

本 题 完整 的 源 代码 详 见 46_Accumulate 项 目 。 


测试 用 例 : 














e 功能 测试 〈 输 入 5、10 求 1 十 2 十 ... 十 5 和 1 十 2 十 ... 十 10) 。 
e 边界 值 测试 (输入 0 和 1) 。 


本 题 考点 : 


e 考查 发 散 思 维 能 力 。 当 习以为常 的 方法 被 限制 使 用 的 时 候 ， 应 
ge 打开 思路 想 出 新 的 办 法 ， 是 能 个 通过 面试 的 关 
e 考查 知识 面 的 广度 和 深度 。 上 面 提供 的 几 种 解法 ， Ce 
数 、 静 态 变 量 、 虚 拟 函 数 、 函 数 指针 、 模 板 类 型 的 实例 化 等 知识 点 。 只 
o aes 才能 在 需要 的 时 候 信 手 牛 来 。 这 就 是 厚 积 薄 
过 程 。 


面试 题 47: 不 用 加 减 乘除 做 加 法 


题目 : ae 数 ， 求 两 个 整数 之 和 ， 要 求 在 函数 体内 不 
得 使 用 十 、 一 、x、= 四 则 运算 符号 。 

aire ene Ga 很 多 人 在 想 : 四 则 运算 都 不 能 用 ， 那 

能 用 什么 啊 ? 可 是 问题 总 是 要 解决 的 ， 我 们 只 能 打开 思路 去 思考 名 种 
Tere. 首先 我 们 可 以 分 析 人 们 是 如 何 做 十 进 制 的 加 法 的 ， 比 如 是 如 何 
得 出 5 十 17 二 22 这 个 结果 的 。 实 际 上 ， 我 们 可 以 分 成 三 步 进 行 : 第 一 步 

a 进位 ， 此 时 相 加 的 结果 是 12 (个 位 数 5 和 7 相 加 不 要 进位 

， 十 位 数 0 和 1 相 加 结果 是 1) ; 第 二 步 做 进位 ，5 十 7 中 有 进位 ， 进 位 
第 三 步 把 前 面 两 个 红 吉 果 加 起 来 ， 12 十 10 的 结果 是 22， 刚 好 5 
十 17 一 22。 

我 们 一 直 在 想 ， 求 两 数 之 和 四 则 运算 都 不 能 用 ， 那 还 能 用 什么 ”对 
数字 做 运算 ， 除 了 四 则 运算 之 外 ， 也 就 只 剩 下 位 运算 了 。 位 运算 是 针对 
二 进 制 的 ， 我 们 就 以 二 进 制 再 来 分 析 一 下 前 面 的 三 步 走 策略 对 二 进 制 是 
不 是 也 适用 。 

5 的 二 进 制 是 101，17 的 二 进 制 是 10001。 还 是 试 着 把 计算 分 成 三 
步 : 第 ee 计 进 位 ， 得 到 的 结果 是 10100 (最 后 一 位 两 个 
on _ 相 加 的 结 二 进 制 的 10。 这 一 步 不 计 进 位 ， 因 此 结果 仍然 
是 0) ; 二 步 记 下 进位 在 这 个 例子 中 只 在 最 后 一 位 相 加 时 产生 一 个 
进位 ， 结果 是 二 进 制 的 10; 第 三 步 把 前 两 步 的 结果 相 加 ， 得 到 的 结果 是 
转换 成 十 进 制 正 好 是 22。 由 此 可 见 三 步 走 的 策略 对 二 进 制 也 是 
4a Yo 
接 下 来 我 们 试 着 把 二 进 制 的 加 法 用 位 运算 来 蔡 代 。 第 一 步 不 考虑 进 

















位 对 每 一 位 相 加 。0 加 0、1 加 1 的 结果 都 0，0 加 1、1 加 0 的 结果 都 是 1。 我 
们 注意 到 ， 这 和 异 或 的 结果 是 一 样 的 。 对 异 或 而 言 ，0 和 0、1 和 1 异 或 的 
结果 是 0， 而 0 和 1、1 和 0 的 异 或 结果 是 1。 接 着 考虑 第 二 步 进 位 ， 对 0 加 
0、0 加 1、1 加 0 而 言 ， 都 不 会 产生 进位 ， 只 有 1 加 1 时 ， 会 向 前 产生 一 个 
进位 。 此 时 我 们 可 以 想象 成 是 两 个 数 先 做 位 与 运算 ， 然 后 再 同 左 移动 一 
位 。 只 有 两 个 数 都 是 1 的 时 候 ， 位 与 得 到 的 结果 是 1， 其 余 都 是 0(。 第 三 
步 把 前 两 个 步骤 的 结果 相 加 。 第 三 步 相 加 的 过 程 依然 是 重复 前 面 两 步 ， 
直到 不 产生 进位 为 止 。 

把 这 个 过 程 想 清楚 之 后 ， 写 出 的 代码 非常 简洁 。 下 面 是 一 段 基于 循 
环 实现 的 参考 代码 : 
int Add(int numl, int num2) 


{ 











int sum, carry; 


do 
sum = numl ^ num2; 
carry = (numl & num2) << 1; 
numl = sum; 
num2 = carry; 
wii te tiina '= Q); 
return numl1; 
} 
源 代 码 : 
本 题 完 整 的 源 代 码 详 见 47_AddTwoNumbers 项 目 。 
测试 用 例 : 
输入 正 数 、 负 数 和 0。 
本 题 考 点 : 


e 考查 发 散 思维 能 力 。 当 十 、 一 、x、=* 运 算 符 都 不 能 使 用 时 ， 应 
人 


e 考 香 对 二 进 制 和 位 运算 的 理解 。 
相关 问题 : 


不 使 用 新 的 变量 ， 交 换 两 个 变量 的 值 。 比 如 有 两 个 变量 a、b， 我 们 
希望 交换 它们 的 值 。 有 两 种 不 同 的 办 法 : 


基于 加 减法 基于 异 或 运算 
a=a+b; a=a* b; 
b= a - b; b= a” b; 
a= a— b; -Wi a) oF 


面试 题 48: 不 能 被 继承 的 类 


题目 ， 用 C++ 设计 一 个 不 能 被 继承 的 类 。 

在 C# 中 定义 了 关键 字 sealed， 被 sealed 修 饰 的 类 不 能 被 继承 。 在 Java 
中 同样 也 有 关键 字 final 表 示 一 个 类 型 不 能 被 继承 。 在 C++ 中 没有 类 似 于 
sealed 和 final 的 关键 字 ， 我 们 只 有 上 自己 来 实现 。 


常规 的 解法 : 把 构造 函数 设 为 私有 函数 


很 多 人 都 能 够 想到 ， 在 C++ 中 子 类 的 构造 冰 数 会 自动 调用 父 类 的 构 
造 函 数 ， 子 类 的 析 构 函数 也 会 自动 调用 父 类 的 析 构 函数 。 要 想 一 个 类 不 
能 锌 继承， 我 们 只 要 把 它 的 构造 函数 和 析 构 函数 都 定义 为 私有 函数 。 那 
么 当 一 个 类 试图 从 它 那 继承 的 时 候 ， 必 然 会 由 于 调用 构造 函数 、 析 构 函 
数 而 导致 编译 错误 。 

可 是 这 个 类 型 的 构造 函数 和 析 构 函数 部 是 私有 函数 ， 我 们 怎样 才能 
得 到 该 类 型 的 实例 呢 ? 我 们 可 以 通过 定义 公有 的 静态 函数 来 创建 和 释放 
类 的 实例 。 基 于 这 个 思路 ， 我 们 可 以 写 出 如 下 代码 : 








class SealedClassl 
{ 
public: 
Static SealedClassl* GetiInstance() 
{ 
return new SealedClassl1(); 


} 


Static void DeleteInstance( SealedClassil* pInstance) 
{ 
delete pInstance; 


} 


private: 
SealedClassl() {} 
~SealedClassl1() {} 


这 个 类 是 不 能 被 继承 ， 但 总 觉得 它 和 普通 的 类 型 有 些 不 一 样 ， 使 用 
ole 比如 我 们 只 能 得 到 位 于 堆 上 的 实例 ， 而 得 不 到 位 于 栈 
JE k 


新 奇 的 解法 : 利用 虚拟 继承 ， 能 给 面试 官 留 下 很 好 的 印象 


` 能 实现 一 个 与 一 般 的 类 型 相 比 除了 不 能 被 继承 之 外 其 他 用 法 都 
一 样 的 类 型 呢 ? 办 法 还 是 有 的 ， 不 过 需要 一 定 的 技巧 。 请 看 如 下 代码 : 
template <typename T> class MakeSealed 


{ 


}; 


friend T; 


private: 
MakeSealed() {} 
~MakeSealed() {} 
}; 


Class SealedClass2 : virtual public MakeSealed<SealedClass2> 
{ 
public: 
SealedClass2() {} 
~SealedClass2() {} 
}; 
这 个 SealedClass2 使 用 起 来 和 一 般 的 类 型 没有 区 别 ， 我 们 可 以 在 栈 


上 、 也 可 以 在 堆 上 创建 实例 。 尽 管 类 MakeSealed<SealedClass2> 的 构造 
函数 和 析 构 函数 都 是 私有 的 ， 但 由 于 类 SealedClass2 是 它 的 友 元 类 型 ， 
此 在 SealedClass2 中 调用 MakeSealed<SealedClass2> 的 构造 函数 和 析 构 
函数 都 不 会 引起 编译 错误 。 

但 当 我 们 试图 从 SealedClass2 中 继承 一 个 类 并 创建 它 的 实例 的 时 
候 ， 却 不 能 通过 编译 。 比 如 我 们 从 SealedClass2 中 继承 出 类 型 Try: 


lass Try : public SealedClass2 





{ 
public: 


由 于 类 SealedClass2 是 从 类 MakeSealed<SealedClass2> 虚 继承 过 来 
的 ， 在 调用 Try 的 构造 函数 的 时 候 ， 会 跳 过 SealedClass2 而 直接 调用 
MakeSealed<SealedClass2> 的 构造 函数 。 非 常 遗 憾 的 是 ，Try 不 是 
MakeSealed<SealedClass2> 的 友 元 类 型 ， 因 此 不 能 调用 它 的 私有 构造 函 
数 。 

通过 上 面 的 分 析 ， 我 们 发 现 从 SealedClass2 继 承 的 类 ， 一 旦 实例 化 
就 会 导致 编译 出 错 ， 因 此 SealedClass2 不 能 被 继承 ， 这 也 就 满足 了 题目 
的 要 求 。 

TE: 第 二 种 方法 的 可 移植 性 不 好 。 虽 然 SealedClass2 在 Visual ”Studio 中 能 够 编译 ， 但 由 于 
GCC 中 对 friend 的 要 求 不 同 于 Visual Studio， 目 前 在 最 新 的 GCC 中 还 不 支持 模板 参数 类 型 作为 友 


元 类 型 。 



























































源 代 码 : 
本 题 完 整 的 源 代 码 详 见 48_SealedClass 项 目 。 
本 题 考 点 : 


e 考查 发 散 思 维 能 力 。 当 要 求 设计 一 个 不 能 被 继承 的 类 时 ， 应 聘 
者 要 号 上 从 把 构造 函数 定义 为 私有 函数 出 发 去 寻找 解 题 方法 。 
© 考 奋 对 C++ 多 个 概念 的 理解 ， 比 如 构造 函数 、 模 板 、 友 元 等 。 


6.6 ”本 章 小 结 


面试 是 我 们 展示 自己 综合 素质 的 时 候 。 除 了 扎实 的 编程 能 力 ， 我 们 
还 需要 表现 自己 的 沟通 能 力 和 学 习 能 力 ， 以 及 知识 迁移 能 力 、 抽 象 建 模 
能 力 和 发 散 思 维 能 力 等 方面 的 综合 实力 〈 如 图 6.3 所 示 ) 。 














图 6.3 应聘 者 的 综合 能 力 的 组 成 





面试 官 对 沟通 能 力 、 学 习 能 力 的 考 租 贯穿 着 面试 的 始终 。 面 试 官 不 
仅 会 留意 我 们 回答 问题 时 的 言语 谈吐 ， 还 会 关注 我 们 是 否 能 抓 住 问题 的 
本 质 从 而 提出 有 针对 性 的 问题 。 通 常 面试 官 认为 善于 提问 的 人 有 较 好 的 
沟通 和 学 习 能 力 。 

知识 迁移 能 力 能 帮助 我 们 轻松 地 解决 很 多 问题 。 有 些 面 试 官 在 提问 
一 道 难 题 之 前 ， 会 问 一 道 相 关 但 比较 简单 的 题目 ， 他 希望 我 们 能 够 从 解 
决 简 单 问题 的 过 程 中 受到 启 太 ， 最 终 解决 较为 复 林 的 问题 。 态 外 ， 我 们 
在 面试 之 前 可 以 做 一 些 练习 。 如 末 面 试 的 时 候 碰 到 类 似 的 题目 ， 残 可 以 
应 用 之 前 的 方法 。 这 要 求 我 们 平时 要 有 一 定 的 积累 ， 并 且 每 做 完 一 道 题 
之 后 都 要 总 结 解 题 方法 。 

有 一 类 很 有 意思 的 面试 题 是 从 日 常生 活 中 提炼 出 来 的 ， 面 试 官 用 这 
种 类 型 的 问题 来 考 但 我 们 抽象 建 模 的 能 力 。 为 了 解决 这 种 类 型 的 问题 ， 
eee oes pera) Sele rear Ul aloe 
Ti tk 0 

有 些 面 试 官 喜 欢 在 面试 的 时 候 限制 使 用 向 规 的 思路 。 这 个 时 候 就 需 
要 我 们 充分 发 挥发 散 思 维 的 能 力 ， 路 出 常规 思路 的 束缚 ， 从 不 同 的 角度 
去 尝试 新 的 办 法 。 

















第 7 章 


在 第 一 章 中 ， 我 们 讨论 








的 项 目 经 历 及 掌握 的 技能 开始 的 。 
介绍 自己 完成 的 工作 (包括 基于 什么 平 








项 目 组 的 贡献 。 


仓 了 面试 的 流程 。 
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接着 进入 重头 戏 技术 面 





两 个 面试 采 例 











通常 一 轮 面试 是 从 面试 官 对 照 着 简历 了 解 应 聘 者 











在 介绍 自己 的 项 目 经 历时 ， 应 聘 者 可 以 参照 STAR 模型 ， 着 重 
了 哪些 技术 、 实 现 了 哪些 算法 等 ) ， 以 及 最 终 对 























I 试 环 节 。 在 这 一 








基础 知识 
到 的 面 


方面 考查 应 聘 者 的 
一 两 个 函数 。 如 果 而 
件 和 错误 处 型 
可 以 尝试 画 
Hy LD 
痢 形成 清晰 的 思路 ， 





















































ERIE 
UR faj A 
等 方面 确保 代码 的 完整 性 和 重 棒 必 
图 让 抽象 的 问题 变 得 形象 化 ， 也 可 以 尝试 举 几 个 
地 解决 小 问题 。 
。 很 多 面试 题 都 不 止 一 种 解决 方案 ， 应 








试 把 大 的 问题 分 解 成 两 个 或 者 多 个 小 问题 再 递归 
从 而 解决 复杂 的 难 





和 〈 详 见 第 2 章 ) ， 
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环节 中 面试 官 会 从 编程 语言 、 
很 有 可 能 会 要 求 应 聘 者 编程 实现 
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应 聘 者 也 不 能 掉以轻心 ， 
E 〈 详 见 第 3 章 ) 。 














题 〈 详 见 第 4 章 ) 
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由 者 可 以 从 时 间 复 杂 度 和 空间 























复杂 度 两 个 方面 选择 最 优 的 解法 〈 详 见 第 5 章 ) 。 














面试 官 除 了 关注 应 聘 者 的 


























当前 招聘 的 项 目 及 其 团队 等 











ij 程 角 
查 应聘 者 的 知识 迁移 能 力 、 才 


薪资 情况 ， A AUST TO 


bE 力 外 ， 他 还 会 关注 应 聘 
象 建 模 能 力 和 发 散 思 维 能 力 
面试 官 会 给 应 聘 者 机 会 
面 提出 几 个 问题 。 不 建议 应 









































式 结果 。 





接 下 来 是 两 个 典型 的 面 








| 试 案 例 ， 我 们 从 中 可 以 直观 





者 的 沟通 
FE 见 第 
问 几 个 最 感 兴趣 的 问题 。 应 聘 者 可 以 从 
聘 者 在 技术 面试 的 时 候 向 面试 官 询问 
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如 果 页 到 的 题目 很 难 ， 应 聘 者 
LL 体 的 例子 去 分 析 隐 舍 


数据 结构 和 算法 等 






































一 定 要 从 基本 功能 、 边 界 条 





的 规律 ， 还 
方法 能 够 帮助 应 聘 

















这 3 和 有 





在 面试 过 程 中 ， 
8 力 和 学 习 能 力 ， 并 有 可 能 


6 章 ) o 














RSS FI 1H 


j 试 的 整个 过 程 。 在 第 一 个 








案例 ( 详 见 7.1 节 〉 中 ， 我 们 将 看 到 面 
EF 见 7.2 节 )〉 中 ， 我 们 将 看 到 面试 官 所 认可 


GE 























的 表现 。 


我 们 希望 应 聘 


1 试 过 程 中 很 多 应 聘 者 都 曾 犯 过 





的 错误 ， 而 在 第 二 个 案例 















































在 面试 过 程 中 充分 表现 
Offer. 


7.1 案例 一 : 








出 自己 的 综合 素质 ， 同 时 也 衷心 祝 








者 能 够 少 犯 甚至 不 犯错 误 TR 





愿 每 个 应 聘 者 都 能 拿 到 自己 心仪 的 





〈 面 试题 49) FEE PG 





Mike: 看 你 简历 上 写 的 是 精通 C/C++i 
应 聘 者 :从 大 一 算 起 的 话 ， 快 六 、 
MRE: 也 是 C/C++ 的 老 程 序 员 了 

















A4 纸 ， 上 下 











有 一 段 打 印 的 代码 ， 如 下 面 所 示 )。 





七 年 了 。 
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你 能 


WE OR) ， 那 先 问 一 个 C++ 的 问题 〈 递 给 应 聘 
能 分 析 一 下 这 段 代 码 的 输出 ? 


ao 这 两 门 语 言 你 用 了 几 年 了 ? 




















者 一 张 


class A 
{ 
private: 
ing wis 
int nZ; 
public: 
A(): n2(0), nl(n2 + 2) 
\ 
void Print () 
{ 
Stas tCouEk <<) "nl: “© ge wl << "2 nas T << nz <= sta: tend: 


}; 


r 


int _tmain(int argc, _TCHAR* argv[]) 
{ 

A a; 

a. Beane (y's 


return 0; 


应 聘 者 : 〈 看 了 一 下 代码 ， 略 作 思 考 ) n1 是 2， 而 n2 是 0。 

面试 官 : 为 什么 ? 

应 聘 者 : 在 构造 函数 的 初始 化 列表 中 ，n2 先 被 初始 化 为 0，n2 的 值 就 是 0 了 。 接 下 来 再 用 
n2 十 2 初始 化 np1， 所 以 n1 的 值 就 是 2。 

[ 注 : 应 聘 者 这 个 问题 的 回答 是 错误 的 ， 详 见 后 面 的 “面试 官 点 评 ”] 

面试 官 ，C++ 是 按照 在 初始 化 列表 中 的 顺序 初始 化 成 员 变 量 的 吗 ? 

应 聘 者 : CEAR) 不 是 这 样 吗 ? 我 不 太 清楚 。 

面试 官 心理 : 

对 成 员 变 量 的 初始 化 顺序 完全 没有 概念 就 号 称 自己 “精通 "C++， 也 太 言 过 其 实 了 。 算 了 ， 
C++ 就 不 接着 问 了 ， 看 看 你 的 编程 能 力 。 
面试 官 ， 没 关系， 我 们 换 一 个 题目 。 能 不 能 介绍 一 下 C 语 言 的 库 函 数 中 atoi 的 作用 ? 
应 聘 者 : atoi 用 来 把 一 个 字符 串 转 换 成 一 个 整数 。 比 如 输入 字符 串 "123"， 它 的 输出 是 数字 










































































123。 
面试 官 : 对 的 。 现 在 就 请 你 写 一 个 函数 StrToInt， 实 现 把 字符 串 转 换 成 整数 这 个 功能 。 当 
然 ， 不 能 使 用 atoi 或 者 其 他 类 似 的 库 函 数 。 你 看 有 没有 问题 ? 
应 聘 者 : (嘴角 出 现 一 丝 自信 的 笑容 〉 没有 问题 。 
































应 聘 者 马上 开始 在 日 纸 上 写 出 了 如 下 代码 : 


int StrToInt (Cliar* string) 


{ 
int number = 0; 
while(*string != 0) 
t 
number = number * 10 + *string - '0'; 


t+string; 
} 


return number; 


应 聘 者 : OTE) 我 已 经 写 好 了 。 

面试 官 心理 : 

我 出 的 题目 有 这 么 简单 吗 ? 你 也 太 小 看 我 了 。 

面试 官 : KAR? (稍微 看 了 看 代码 ) 你 觉得 这 代码 有 没有 问题 ? 仔细 检查 一 下 看 看 。 

应 聘 者 : “(从头 开始 读 代码 ) 哦 ， 不 好 意思 ， 忘 了 检查 字符 串 是 空 指 针 的 情况 。 

应 聘 者 拿 起 笔 ， 在 原来 代码 上 添加 两 行 新 的 代码 。 修 改 之 后 的 代码 
如 下 : 


int StrToInt(char* string) 


{ 











if(string == NULL) 
return 0; 


int number = 0; 
while(*string != 0) 
{ 
number = number * 10 + *string - '0'; 


++string; 


} 


return number; 
} 


面试 官 : 改 好 了 ? (看 了 一 下 新 的 代码 〉 当 字符 串 为 空 的 时 候 ， 你 的 返回 是 9。 如 果 输 入 
的 字符 串 是 "0" 的 时 候 ， 返 回 是 什么 ? 
应 聘 者 : 也 是 0。 


面试 官 : 两 种 情况 都 得 到 返回 值 0， 那 么 当 这 个 函数 的 调用 者 得 到 返回 值 0 的 时 候 ， 他 怎 
么 知道 是 哪 种 情况 ? 


应 聘 者 : 〈 脸 上 表情 有 些 困 惑 ) 不 知道 
面试 官 : 你 知道 atoj 是 怎么 区 分 的 吗 ? 


























应 聘 者 : 《努力 回忆 ， 有 些 慨 张 ) 不 记得 了 。 
面试 官 : atoi 是 通过 一 个 全 局 变量 来 区 分 的 。 如 果 是 非法 输入 ， 返 回 0 并 把 这 个 全 局 变量 
设 为 一 个 特殊 标记 。 如 果 输 入 是 "0"， 则 返回 0， 不 会 设置 全 局 变量 。 这 样 当 atoi 的 调用 者 得 到 返 
回 值 0 的 时 候 ， 可 以 通过 检查 全 局 变量 得 知 输入 究竟 是 非法 输入 还 是 字符 串 "0"。 

应 聘 者 : 哦 。〈 拿 起 笔 准 备 写 代 码 ) 我 马上 修改 。 

面试 官 等 一 下 ， 除 了 空 字符 串 之 外 ， 还 有 没有 可 能 有 其 他 类 型 的 非法 输入 ? 

应 聘 者 : 《陷入 思考 ， 额 头 上 出 现 汗 珠 ) 如 果 字 符 串 中 含有 '0' 到 '9' 之 外 的 字符 ， 那 么 这 样 
的 输入 也 是 非法 的 。 

面试 官 : 所 有 '0 到 '9 之 外 的 字符 都 是 非法 的 吗 ? 

应 聘 者 : 加 号 和 减 号 应 该 也 是 合法 的 输入 字符 。 

面试 官 : 对 的 。 先 好 好 想 想 ， 想 清楚 了 再 开始 写 代 码 。 

应 聘 者 思考 几 分 钟 之 后 ， 写 下 了 如 下 代码 : 





































































































enum Status {kValid = 0, kInvalid}; 
int g nStatus = kValid; 


int StrToInt(const char* str) 


{ 


g nStatus kinvalid; 


tite: = Ox 


if (str != NULL) 
{ 


const char* digit = str; 


bool minus = false; 
if(*digit == "+") 
digit: +t; 
else if(*digit == '-') 
{ 
digit tt? 
minus = true; 
} 


while(*digit != '\0') 

{ 
if(*digit >= '0' && *digit <= '9") 
{ 


num = num * 10 + (*digit = '0"'); 
digit++; 
} 
else 
{ 
num = 0; 
break; 
} 
} 
if(*digit == "NO 


{ 
g nStatus = kValid; 
if (minus) 
num = 0 - num; 


} 


return num; 


面试 官 : (看 到 应 聘 者 写 完了 )〉 能 不 能 简要 地 解释 一 下 你 的 代码 ? 

应 聘 者 : 我 定义 了 一 个 全 局 变量 g_Status 来 标记 是 不 是 遇 到 了 非法 输入 。 如 果 输 入 的 字符 
串 指针 是 空 指针 ， 则 标记 该 全 局 变量 然后 直接 返回 。 接 下 来 我 开始 忆 有 历 字符 串 中 的 所 有 字符 。 
由 于 正 负 号 只 有 可 能 出 现在 字符 串 的 第 一 个 字符 ， 我 们 先 处理 字 符 串 的 第 一 个 字符 。 如 果 第 一 

















Sal 






























































个 字符 是 符号 ， 则 标记 当前 的 数字 是 负数 ， 并 在 最 后 确保 返回 值 是 负数 。 在 处 理 后 续 字 符 时 ， 
当 遇 到 '0' 到 '9' 之 外 的 字符 时 ， 终 止 遍历 。 如 果 遇 到 了 数字 ， 则 把 数值 累加 上 去 。 
面试 官 心理 : 



































这 段 代 码 已 经 写 得 不 错 ， 你 的 编程 能 力 看 起 来 还 不 错 ， 只 是 编程 的 习惯 不 太 好 ， 不 会 在 
编码 之 前 想 好 可 能 有 哪些 输入 ， 从 而 在 代码 中 留 下 太 多 的 漏洞 。 这 次 修改 的 几 个 问题 都 是 我 已 
经 提醒 你 的 ， 没 有 提醒 的 你 自己 没有 找 出 一 个 。 

面试 官 ， 不 错 。 觉 得 功能 上 还 有 什么 遗漏 吗 ? 

应 聘 者 ， 还 有 遗漏 ? (思索 良久 ) 要 不 要 考虑 溢出 ? 

面试 官 ， 你 觉得 呢 ? 如 果 输 入 的 是 一 个 空 字符 串 "， 你 觉得 应 该 输出 什么 ? 

应 聘 者 : "不 是 个 数字 ， 我 想 应 该 是 返回 9， 同 时 把 g_nStatus 设 为 非法 输入 。 
面试 官 ， 那 你 能 分 析 一 下 你 现在 的 输出 是 什么 吗 ? 

应 聘 者 ，《〈 紧 张 ， 声 音 有 些 发 持 ) 好 像 返 回 值 是 0， 但 没有 设置 g_nstatns 为 非法 答 入 ? 

面试 官 ， 喝 。 我 们 再 考虑 一 些 有 意思 的 输入 ， 比 如 输入 的 字符 串 只 有 一 个 正 号 或 者 负 
号 ， 你 期 待 的 输出 是 什么 ? 

应 聘 者 :如 果 只 有 一 个 正 号 或 者 负 号 ， 后 面 没有 跟着 数字 ， 我 想 也 不 是 有 效 的 输入 。 
(开始 分 析 代 码 ) 我 的 返回 值 是 0， 但 不 会 设置 g_nStatus。 

面试 官 ， 由 于 时 间 也 差不多 了 ， 已 经 没有 时 间 给 你 再 做 修改 了 。 我 的 问题 问 完了 ， 你 有 
什么 问题 需要 问 我 的 吗 ? 

应 聘 者 :你 们 公司 工资 待遇 怎么 样 ? 

面试 官 ， 你 的 期 望 值 是 多 少 呢 ? 

应 聘 者 ;我 有 不 少 同学 的 月 薪 税 前 超过 8000， 我 不 想 低 于 他 们 。 

面试 官 心理 : 

我 不 是 HR， 别 和 我 谈 工 资 。 

面试 官 ; 我 们 公司 由 人 事 部 门 统一 确定 应 届 毕 业 生 的 工资 ， 所 以 你 的 这 个 问题 我 不 能 直 
接 回答 ， 不 过 你 的 期 望 值 我 倒 可 以 转告 HR。 

应 聘 者 ;好 的 ， 谢 谢 。 

面试 官 ， 还 有 其 他 问题 吗 ? 

应 聘 者 ， 没有 了 。 

面试 官 ， 那 这 轮 面试 就 到 这 里 结束 吧 。 


面试 官 点 评 : 
这 名 应 聘 者 在 简历 中 写 他 精通 C/C++， 本 来 我 对 他 的 表现 是 充满 了 




























































































































































































期 竺 的。 但 在 他 回答 错 了 第 一 个 C++ 的 语法 题 之 后 ， 他 给 我 留 下 的 印象 
就 不 是 很 好 了 。 这 束 是 希望 越 大 失望 越 大 吧 。 实 际 上 ， 构 造 函 数 的 初始 
化 列表 是 C++ 中 经 第 使 用 的 一 个 概念 。 在 C++ 中 ， 成 员 变 量 的 初始 化 顺 
序 只 与 它们 在 类 中 声明 的 顺序 有 关 ， 而 与 在 初始 化 列表 中 的 顺序 无 关 。 
在 前 面 的 问题 中 ，n1 先 于 n2 被 声明 ， 因 此 nl 也 会 在 n2 之 前 被 初始 化 ， 所 
以 我 们 先 会 用 n2 十 2 去 初始 化 n1。 由 于 n2 这 个 时 候 还 没有 被 初始 化 ， 
此 它 的 值 是 随机 的 。 用 此 时 的 n2 加 上 2 去 初始 化 n1，nl 的 值 只 是 一 个 随 
机 值 。 接 下 来 再 用 0 初始 化 n2， 因 此 最 终 n2 的 值 是 0。 

接 下 来 是 要 求 应 聘 者 把 一 个 字符 串 转 换 成 整数 ， 这 看 起 来 是 道 很 简 
单 的 题目 ， 实 现 其 基本 功能 ， 大 部 分 人 都 能 用 10 行 之 内 的 代码 解决 。 可 
是 ， 当 我 们 要 把 很 多 特殊 情况 即 测试 用 例 都 考虑 进去 ， 却 不 是 一 件 容易 
的 事情 。 解 决 数值 转换 问题 本 身 不 难 ， 但 我 希望 在 写 转换 数值 的 代码 之 
前 ， 应 聘 者 至 少 能 把 空 指针 NULL、 空 字符 串 ""、 正 负 号 、 洲 出 等 方 方 
面 面 的 测试 用 例 都 考虑 到 ， 并 在 写 代 码 的 时 候 对 这 些 特殊 的 输入 部 定义 
好 合理 的 输出 。 当 然 ， 这 些 输出 并 不 一 定 要 和 atoi 完 全 保持 一 致 ， 但 必 
须要 有 显 式 的 说 明 ， 和 面试 官 沟通 好 。 

这 个 应 聘 者 最 大 的 问题 就 是 还 没有 养 成 在 写 代 码 之 前 考 碟 所 有 可 能 
的 测试 用 例 的 习惯 ， 远 辑 不 够 严谨 ， 因 此 一 开始 的 代码 只 处 理 了 最 基本 
的 数值 转换 。 后 来 我 每 次 提醒 他 一 处 特殊 的 测试 用 例 之 后 ， 他 改 一 处 代 
码 。 尺 管 他 已 经 做 了 两 次 修改 ， 但 仍然 有 不 少 很 明显 的 漏洞 ， 特 殊 输 入 
空 字符 串 ""， 边 界 条 件 比 如 最 大 的 正 整 数 与 最 小 的 负 整 数 等 。 由 于 这 道 
题 思 路 本 映 不 难 ， 因 此 我 希望 他 能 把 问题 考虑 得 尽 可 能 周到 ， 代 码 尽量 
写 完整 。 下 面 是 参考 代码 : 


























enum Status {kValid = 0, kInvalid}; 
int g nStatus = kValid; 


int StrToInt(const char* str) 
{ 
g nStatus = kinvalid; 
long long num = 0; 


1f (str I= NULL && *str t= EVO”) 
{ 
bool minus = false; 
LE == sF") 
SGr EEs 
else if(*str == '-') 
{ 
SE Asks 
minus = true; 


} 


ES A= "NO 
{ 
num = StrToIntCore(str, minus); 
} 
} 


return (int) num; 


long long StrToIntCore(const char* digit, bool minus) 


{ 


long long num = 0; 


while(*digit != '\0"') 
{ 
if (*digit >= "0° && *digit <= "9") 
{ 
int flag = minus ? -1 : 1; 
num = num * 10 + flag * (*digit - '0'); 


if((!minus && num > Ox7FFFFFFF) 
| (minus && num < (signed int) 0x80000000) ) 


在 前 面 的 代码 中 ， 把 空 字 符 串 "" 和 只 有 一 个 正 号 或 者 负 号 的 情况 都 
考虑 到 了 。 同 时 正 整 数 的 最 大 值 是 0x7FFF FFFF， 最 小 的 负 整 数 是 
0x8000 ”0000， 因 此 我 们 需要 分 两 种 情况 来 分 别 判断 整数 是 否 及 生 上 注 
出 或 者 下 洲 出 。 

最 后 他 在 提问 环节 给 我 留 下 的 印象 不 是 很 好 。 他 只 有 一 个 问题 ， 是 
关于 薪水 方面 。 是 不 是 反映 出 他 找 工 作 仅 仅 关 心 工 资 ? 通常 只 关心 工资 
待遇 的 员工 是 非常 容易 流失 的 ， 而 且 他 把 期 望 值 设 在 8000 的 唯一 理由 是 
他 有 同学 的 工资 超过 这 个 数 。 他 对 自己 有 没有 一 个 定位 ? 
虽然 从 他 后 来 改写 代码 的 过 程 来 看 ， 他 的 编程 能 力 还 是 不 错 的 ， 但 

















我 担心 以 他 现在 的 编程 习惯 ， 由 于 没有 一 个 全 面 的 考 谍 ， 他 号 出 的 代码 
将 会 漏洞 百出 ， 和 鲁 棒 性 也 得 不 到 保证 。 总 的 来 说 ， 我 的 意见 是 我 们 不 能 
录取 这 名 应 聘 者 。 


源 代码 : 
本 题 完整 的 源 代码 详 见 49_StringToInt 项 目 。 
测试 用 例 : 


e 功能 测试 (输入 的 字符 串 表示 正 数 、 负 数 和 0) 。 

e 边界 值 测试 〈 最 大 的 正 整数 、 最 小 的 负 整 数 ) 。 

特殊 输入 测试 〈 输 入 字符 串 es 输入 字符 串 ARF 
符 串 、 输 入 的 字符 串 中 有 非 数 字 字 符 等 ) 


7.2 ”案例 二 : (面试 题 50) 树 中 两 个 结 点 的 最 低 
公共 祖先 


面试 官 ， 前 面 两 轮 面试 下 来 感觉 怎么 样 ? 
应 聘 者 : 感觉 还 好 ， 只 是 大 脑 连续 转 了 两 个 小 时 ， 有 点 累 。 
面试 官 : 面试 是 个 体力 活 ， 是 挺 累 的 。 不 过 程序 员 这 个 行当 本 身 也 是 体力 活 ， 没 有 好 的 
身体 还 真 撑 不 住 。 面 试 中 也 要 看 看 你 们 来 应 聘 的 体力 怎么 样 。 
WHE: (RAF AAS) 说 得 有 道理 。 
面试 官 : 开 个 玩笑 。 现 在 可 以 开始 面试 了 吧 ? 
WHE: 好 的 ， 我 准备 好 了 。 
面试 官 : 请 简要 介绍 一 下 你 最 近 的 一 个 项 目 。 
MSH: 我 最 近 完 成 的 项 目 是 Civil 3D (一 款 基 于 AutoCAD 的 土木 设计 软件 ) 中 的 Multi- 


Target。 这 个 Target 指 的 是 道路 的 边缘 。 之 前 道路 的 边缘 只 能 是 Civil 中 的 一 个 数据 类 型 叫 
Sk 我 的 工作 是 让 Civil 文 持 其 他 类 型 的 数据 做 道路 的 边缘 ， 比 如 AutoCAD 中 的 Polyline 












































































































































面试 官 ， 有 没有 考虑 到 以 后 有 可 能 会 添加 新 的 数据 类 型 作为 道路 的 边缘 ? 


ARE: 这 个 在 开 及 的 过 程 中 就 发 生 过 。 在 第 二 版 的 需求 文档 中 添加 了 一 种 叫做 Pipeline 
的 道路 边缘 。 由 于 我 的 设计 中 考虑 了 扩展 性 ， 最 后 只 要 添加 新 的 class 就 行 了 ， 几 乎 不 需要 对 已 
有 的 代码 做 任何 修改 。 


面试 官 : 你 是 怎么 做 到 的 ? 

应 聘 者 在 白 纸 上 用 UML 画 了 一 张 类 型 关系 图 〈 图 略 ) 。 

应 聘 者 : 〈 指 着 图 解释 ) 从 这 张 图 我 们 可 以 看 出 ， 一 旦 需要 支持 新 的 道路 边缘 比如 
Pipeline， 我 们 只 需 继承 出 新 的 class 就 可 以 了 ， 对 已 有 的 其 他 class 没 有 影响 。 


面试 官 心理 : 







































































对 自己 的 工作 讲 得 很 细致 、 很 深入 ， 这 个 项 目的 确 是 你 设计 和 实现 的 。 看 得 出 来 ， 你 对 
面向 对 象 的 设计 和 开发 有 着 较 深 的 理解 。 


MAR: 《点 点 头 ) 的 确 是 这 样 的 。 接 下 来 我 们 做 一 个 编程 题 吧 。 我 的 题目 是 输入 两 个 
树 结 点 ， 求 它们 的 最 低 公共 祖先 。 


应 聘 者 : 这 树 是 不 是 二 又 树 ”? 

面试 官 : 是 又 怎么 样 ， 不 是 又 怎么 样 ? 

应 聘 者 : 如 果 是 二 又 树 ， 并 且 是 二 又 搜索 树 ， 是 可 以 找到 公共 结 点 的 。 
面试 官 ， 那 假设 是 二 叉 搜索 树 ， 你 怎么 查找 呢 ? 


we: (有 些 激动 ， 说 得 很 快 ) 二 叉 搜 索 树 是 排序 过 的 ， 位 于 左 子 树 的 结 点 都 比 父 结 
点 小 ， 而 位 于 右 子 树 的 结 点 都 比 父 结 点 大 ， 我 们 只 需要 从 树 的 根 结 点 开始 和 两 个 输入 的 结 点 进 
行 比较 。 如 果 当 前 结 点 的 值 比 两 个 结 点 的 值 都 大 ， 那 么 最 低 的 共同 父 结 点 一 定 是 在 当前 结 点 的 
左 子 树 中 ， 于 是 下 一 步 吉 历 当前 结 点 的 左 子 结 点 。 如 果 当 前 结 皮 的 值 比 两 个 结 点 的 值 都 小 ， 那 
么 最 低 共同 父 结 点 一 定 在 当前 结 点 的 右 子 树 中 ， 于 是 下 一 步 通 历 当前 结 点 的 右 子 结 点 。 这 样 在 
树 中 从 上 到 下 找到 的 第 一 个 在 两 个 输入 结 点 的 值 之 间 的 结 点 ， 就 是 最 低 的 公共 祖先 。 


面试 官 ， 看 起 来 你 对 这 个 题目 很 熟悉 ， 是 不 是 以 前 做 过 啊 ? 
应 聘 者 : CREED) 这 个 .…… 碰 巧 .……. 


面试 官 : (KR) 那 咀 们 把 题目 稍微 换 一 下 。 如 果 这 棵 树 不 是 二 又 搜 索 树 ， 甚 至 连 二 又 树 
都 不 是 ， 而 只 是 普通 的 树 ， 又 该 怎么 办 呢 ? 


应 聘 者 ，《〈 停 下 来 想 了 十 几 秒 ) 树 的 结 点 中 有 没有 指向 父 结 点 的 指针 ? 
面试 官 心理 : 

反应 挺 快 的 ， 而 且 提 的 问题 针对 性 很 强 。 你 的 沟通 能 力 不 错 。 
面试 官 ， 为 什么 需要 指向 父 结 点 的 指针 ? 

应 聘 者 在 白 纸 上 画 了 一 张 图 ， 如 图 7.1 所 示 。 
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图 7.1 树 中 的 结 点 有 指向 父 结 点 的 指针 ， 用 虚线 箭头 表示 

应 聘 者 : 〈 指 着 自己 画 的 图 7.1 解 释 ) 如 果树 中 的 每 个 结 点 《除根 结 点 之 外 ) 都 有 一 个 指 

癌 父 结 点 的 指针 ， 这 个 问题 可 以 转换 成 求 两 个 链表 的 第 一 个 公共 结 点 。 假 设 树 结 点 中 指向 父 结 
点 的 指针 是 pParent， 那 么 从 树 的 每 一 个 叶 结 点 开始 都 有 一 个 由 指针 pParent 串 起 来 的 链表 ， 这 些 
































链表 的 尾 指针 都 是 树 的 根 结 点 。 输 入 两 个 结 点 ， 那 么 这 两 个 结 点 位 于 两 个 链表 上 ， 它 们 的 最 低 
公共 祖先 刚好 就 是 这 两 个 链表 的 第 一 个 公共 结 点 。 比 如 输入 的 两 个 结 点 分 别 为 F 和 HH， 那 么 F 在 
链表 F->D->B->A 上， 而 H 在 链表 H->E->B->A 上 ， 这 两 个 链表 的 第 一 个 交点 B 刚 好 也 是 它们 的 最 
低 公 共 祖 先 。 
面试 官 : 求 两 个 链表 的 第 一 个 共同 结 点 这 个 题目 你 是 不 是 之 前 也 做 过 
应 聘 者 : GARUNA, IMME) XA ee 又 被 你 发 现 了 ..….. 


面试 官 心理 : 


能 够 把 这 个 题目 转换 成 求 两 个 链表 的 第 一 个 公共 结 点 ， 你 的 知识 迁移 能 力 不 错 。 感 觉 你 
aA. 基本 上 达到 录用 标准 了 。 不 过 我 很 有 兴趣 看 看 你 的 极限 在 哪里 。 再 加 大 点 
难度 试 试 吧 


面试 官 : 上 
树 中 的 结 点 没有 指向 父 结 点 的 指针 。 


应 聘 者 : 〈 稍 微 流 露出 一 丝 抓 狂 的 表情 ， 语 气 中 透 出 失望 ) 好 吧 ， 我 再 想 想 。 
面试 官 : 这 个 题目 也 只 比 前 面 的 两 个 稍微 难 一 点 点 ， 你 能 搞定 的 。 


应 聘 者 : 〈 静 下 来 思考 了 两 分 钟 ) 所 谓 两 个 结 点 的 公共 祖先 ， 指 的 是 这 两 个 结 点 都 出 现 
在 某 个 结 点 的 子 树 中 。 我 们 可 以 从 根 结 点 开始 衣 历 一 棵 树 ， 每 过 历 到 一 个 结 点 时 ， 判 断 两 个 输 
入 结 点 是 不 是 在 它 的 子 树 中 。 如 果 在 子 树 中 ， 则 分 别 通 历 它 的 所 有 子 结 点 ， 并 判断 两 个 输入 结 
点 是 不 是 在 它们 的 子 树 中 。 这 样 从 上 到 下 一 直 找到 的 第 一 个 结 点 ， 它 自己 的 子 树 中 同时 包含 两 
E 寺 点 却 没 有 ， 那 么 该 结 点 就 是 最 低 的 公共 祖先 。 


面试 官 : 能 不 能 举 个 具体 的 例子 说 明 你 的 思路 ? 

应 聘 者 : 〔 一 边 在 纸 上 画 图 7.2 一 边 解释 ) 假设 还 是 输入 结 点 FE 和 互 。 我 们 先 判 断 A 的 子 树 
中 是 否 同时 包含 结 点 FE 和 H， 得 到 的 结果 为 tue。 接 着 我 们 再 先后 判断 A 的 两 个 子 结 点 B 和 C 的 子 
树 是 不 是 同时 包含 FP 和 HH， 结果 是 B 的 结果 是 true 而 C 的 结果 是 false 。 接 下 来 我 们 再 判断 B 的 两 个 
le aia 才 果 都 是 false。 于 是 B 是 最 后 一 个 公共 祖先 ， 即 我 们 的 输 


























































































































































































































































































































图 7.2 ”一 棵 普通 的 树 ， 树 中 的 结 点 没有 指向 父 结 点 的 指针 


面试 官 : 听 起 来 不 错 。 很 明显 ， 当 我 们 判断 以 结 点 A 为 根 的 树 中 是 否 含 有 结 点 F 的 时 候 ， 
我 们 需要 对 D、 | Wb; 接 下 来 判断 以 结 点 B 为 根 的 树 中 是 否 含有 结 点 FE 的 时 候 ， 我 们 
还 是 需要 对 D、E 等 结 点 再 过 历 一 过 。 这 种 思路 会 对 同一 结 点 重复 志 历 很 多 次 。 你 想 想 看 还 有 没 
有 更 快 的 算法 ? 































































































应 聘 者 : 〈 双 肘 抵 住 桌子 ， 双 手 抱 住 头 项， 兰若 思索 两 分 钟 ) 可 以 用 辅助 内 存 吗 ? 
面试 官 : 你 需要 多 大 的 辅助 内 存 ? 

应 聘 者 : 我 的 想法 是 用 两 个 链表 分 别 保 存 从 根 结 点 到 输入 的 两 个 结 点 的 路 径 ， 然 后 把 问 
题 转换 成 两 个 链表 的 最 后 公共 结 点 。 
面试 官 : CAA, MERT) 咽 ， 具 体 说 说 。 

应 聘 者 : 我 们 首先 得 到 一 条 从 根 结 点 到 树 中 某 一 结 点 的 路 径 ， 这 就 要 求 在 遍历 的 时 候 ， 
有 一 个 辅助 内 存 来 保存 路 径 。 比 如 我 们 用 前 序 遍 历 的 方法 来 得 到 从 根 结 点 到 HH 的 路 径 的 过 程 是 
这 样 的 (1) 遍历 到 A， 把 A 存放 到 路 径 中 去 ， 路 径 中 只 有 一 个 结 点 A; (2) WPB, EBF 
到 路 径 中 去 ， 此 时 路 径 为 A->B; G) 遍历 到 D， 把 D 存 放 到 路 径 中 去 ， 此 时 路 径 为 A->B->D; 
(4) 忆 历 到 FE， 把 F 存 放 到 路 径 中 去 ， 此 时 路 径 为 A->B->D->F; (5) F 已 经 没有 子 结 点 了 ， 因 
此 这 条 路 径 不 可 能 到 达 结 点 英 。 把 F 从 路 径 中 删除 ， 变 成 A->B->D; (6) 遍历 G。 和 结 点 F 一 
样 ， 这 条 路 径 也 不 能 到 达 HH。 人 遍历 完 G 之 后 ， 路 径 仍 然 是 A->B->D; (7) 由 于 D 的 所 有 子 结 点 都 
遍历 过 了 ， 不 可 能 到 达 结 点 H， 因 此 DD 不 在 从 A 到 H 的 路 径 中 ， 把 D 从 路 径 中 删除 ， 变 成 A->B; 
(8) 遍历 E， 把 E 加 入 到 路 径 中 ， 此 时 路 径 变 成 A->B->E， (9) 遍历 H， 己 经 到 达 目 标 结 点 ， 
A->B->E 就 是 从 根 结 点 开始 到 达 H 必 须 经 过 的 路 径 。 

Mite: 然后 呢 ? 

应 聘 者 : 同样 ， 我 们 也 可 以 得 到 从 根 结 点 开始 到 达 F 必 须 经 过 的 路 径 是 A->B->D。 接 着 ， 
我 们 求 出 这 两 个 路 径 的 最 后 公共 结 点 ， 也 就 是 B。B 这 个 结 点 也 是 F 和 HH 的 最 低 公 共 祖 先 。 
面试 官 : 这 种 思路 的 时 间 和 空间 效率 是 多 少 ? 

应 聘 者 : 为 了 得 到 从 根 结 点 开始 到 输入 的 两 个 结 点 的 两 条 路 径 ， 需 要 遍历 两 次 树 ， 每 遍 
历 一 次 的 时 间 复 杂 度 是 O(n) 。 得 到 的 两 条 路 径 的 长 度 在 最 差 情况 时 是 O Cn) ， 通 常情 况 下 两 
条 路 径 的 长 度 是 DO Cogn) 。 


面试 官 心理 : 

显然 ， 你 对 数据 结构 的 理解 比 大 多 数 人 要 深刻 得 多 ， 期 待 你 的 代码 。 
Mine: WR, AA) 不 错 。 根 据 这 个 思路 写 出 C/C++ 代码 ， 怎 么 样 ? 
应 聘 者 : 好 的 ， 没 问题 。 

应 聘 者 先后 下 了 三 个 函数 : 


























































































































































































































bool GetNodePath (TreeNode* pRoot, TreeNode* pNode, list<TreeNode*>é 


path) 
{ 
if (pRoot == pNode) 
return true; 


path.push_ back (pRoot); 
bool found = false; 


vector<TreeNode*>::iterator i = pRoot->m_vChildren.begin(); 
while (!found && i < pRoot-—>m_vChildren.end() ) 


{ 
found = GetNodePath(*i, pNode, path); 


++i; 
} 


if (! found) 
path.pop_back(); 


return found; 
} 


TreeNode* GetLastCommonNode 

( 
const list<TreeNode*>é pathi, 
const list<TreeNode*>é& path2 


pathi.begin(); 


list<TreeNode*>::const_iterator iteratorl 
path2.begin(); 


list<TreeNode*>::const_iterator iterator2 


TreeNode* pLast = NULL; 


while (iteratorl != pathi.end() && iterator2 != path2.end()) 
{ 
if (*iteratorl == *iterator2) 
pLast = *iteratorl1; 
iteratori++; 
iterator2++; 


} 


return pLast; 


} 


TreeNode* GetLastCommonParent (TreeNode* pRoot, TreeNode* pNodel, 


TreeNode* pNode2) 
{ 
if(pRoot == NULL || pNodel == NULL || pNode2 == NULL) 
return NULL; 


list<TreeNode*> pathl; 
GetNodePath(pRoot, pNodel, pathi) 


se 


list<TreeNode*> path2; 
GetNodePath(pRoot, pNode2, path2); 


return GetLastCommonNode(pathl, path2); 








应 聘 者 : 代码 中 GetNodePath 用 来 得 到 从 根 结 点 pRoot 开 始 到 达 结 点 PNode 的 路 径 ， 这 条 路 
中 。 函 数 LastCommonNode 用 来 得 到 两 个 路 径 pathl1 和 path2 的 最 后 一 个 公共 结 k 
函数 GetLastCommonParent 先 调用 GetNodePath 得 到 pRoot 到 达 pNodel 的 路 径 path1， 再 得 SilpRoot 
到 达 pNode2 的 路 答 path2 ， 接 着 调用 GetLastCommonPath 得 到 path1 和 path2 的 最 后 一 个 公共 结 点 ， 
即 我 们 要 找 的 最 低 公 共 祖 先 。 
站 
;的 吗 ? 
应 聘 者 : EEZ) 我 想 问 问 关 于 项 目 合作 的 事情 。 你 们 项 目 组 和 美国 总 部 的 同事 是 
怎么 合作 的 ? 是 天 国信 做 好 设计 ， 然后 交 给 中 国 这 边 做 具 体 实现 吗 ? 


面试 官 : 理论 上 说 中 国 同事 与 美国 同事 之 间 的 合作 是 平等 的 ， 不 全 是 美国 说 了 算 。 我 们 
国 的 团队 也 有 自己 的 项 目 经 理 做 产品 设计 。 现 在 两 边 的 团队 都 有 自己 负责 的 功能 。 只 是 我 们 
的 团队 成 员 和 美国 同事 比 起 来 ， 经 验 还 不 够 ， 对 产品 的 理解 没有 美国 同事 那么 深刻 。 因 此 当 两 
边 意 见 出 现 分 卜 的 时 候 ， 他 们 的 意见 更 能 得 到 上 层 的 重视 。 

应 聘 者 : 两 边 的 团队 都 有 哪些 沟通 的 方式 ? 


面试 官 : 平时 做 相关 工作 的 同事 之 间 会 有 大 量 的 E-mail 交 流 。 每 周二 的 早上 《美国 时 间 是 
星期 一 的 下 午 ) 我 们 有 一 个 例会 ， 所 有 同事 都 会 参加 。 在 会 上 大 家 会 讨论 项 目的 进度 、 遇 到 的 
困难 等 事项 。 男 外 ， 由 于 最 近 我 们 这 边 招 了 不 少 新 员工 ， 美 国 那 边 正 计 划 选 派 一 个 资深 的 工程 
师 过 来 给 新 员工 做 培训 。 


应 聘 者 : 那 中 国 这 边 的 员工 有 机 会 去 美国 吗 ? 


面试 官 : eR 让 新 员工 在 两 年 之 内 至 少 有 机 会 去 美国 接受 
一 次 培训 ， 以 熟悉 公司 总 部 的 文化 。 只 是 最 近 由 于 大 的 经 济 环境 不 是 很 好 ， 公 司 在 严格 控制 差 
旅费 用 ， 因此 这 个 项 目的 执行 受到 了 一 -点 影响 。 还 有 其 他 问题 吗 ? 


应 聘 者 : 〈 想 了 一 会 儿 ) RAT. 

面试 官 心理 : 

从 最 后 几 个 问题 可 以 看 出 ， 你 对 我 们 的 项 目 和 团队 很 有 兴趣 。 同 样 ， 我 也 希望 你 能 加 入 
我 们 的 团队 一 起 做 项 目 。 这 轮 面 试 你 通过 了 。 
面试 官 ， 由 于 时 间 关 系 ， 这 轮 面 试 就 到 这 里 ， 怎 么 样 ? 
应 聘 者 : CRRIL, MEPER) 谢谢 。 


面试 官 点 评 


求 树 中 两 个 结 反 的 最 低 公 共和 祖先 ， 不 能 说 只 是 一 个 题目 ， 而 应 该 说 
是 一 组 题目 ， 不 同 条 件 下 的 题目 是 完全 不 一 样 的。 一 开始 的 时 候 ， 我 有 
意 没 有 说 明 树 的 特点 ， 比 如 树 是 不 是 二 又 树 、 树 中 的 结 点 是 不 是 有 一 个 
指 癌 父 结 点 的 指针 。 我 把 题目 说 得 模棱两可 是 希望 应 聘 者 能 够 主动 向 我 
提出 问题 ， 一 步 一 步 卉 清 我 的 意图 。 如 条 一 个 应 聘 者 能 够 在 面试 过 程 中 
主动 问 出 高 质量 的 问题 以 并 清楚 题目 的 要 求 ， 我 会 沉 得 他 态度 积极 并 且 
具有 较 强 的 沟通 能 

在 这 轮 面 试 中 ， 该 应 聘 者 表现 得 比较 积极 主动 。 一 开始 听 到 题目 之 
后 ， 他 马上 询问 我 树 是 不 是 二 又 树 。 在 我 答复 可 以 是 二 又 树 之 后 他 立即 
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给 出 了 当 树 是 二 义 排 序 树 时 的 解法 。 我 看 出 了 他 之 前 做 过 这 个 问题 ， 于 
征 就 把 题目 的 和 要求 设 为 树 只 是 普通 的 树 而 不 一 定 是 二 又 树 。 他 的 反应 很 
快 ， 立 即 叉 问 我 树 中 的 结 点 有 没有 指 问 父 结 点 的 指针 。 在 第 二 个 问题 得 
到 肯定 的 答复 之 后 ， 他 把 问题 转换 成 求 两 个 链表 的 第 一 个 公共 结 点 。 他 
这 段 的 表现 很 好 ， 问 的 两 个 问题 都 很 有 针对 性 ， 表 明 他 对 这 种 类 型 的 问 
题 有 很 深 的 理解 ， 给 我 留 下 了 很 好 的 印象 。 

通常 面试 的 时 候 让 应 聘 者 写 出 有 指 同 父 结 反 的 指针 这 种 情况 的 代码 
也 就 差不多 了 ， 但 考虑 到 他 之 前 做 过 类 似 的 问题 ， 同 时 我 觉得 他 反应 很 
快 ， 功 确 不 错 ， 以 他 的 能 力 应 该 可 以 挑战 一 下 更 局 的 难度 。 于 是 我 接 下 
来 把 指 回 父 结 反 的 指针 去 挥 ， 决 定 再 加 大 难度 测试 一 下 他 的 水 平 到 确 有 
多 深 。 他 再 一 次 表现 出 很 快 的 反应 能 力 ， 思 考 了 一 两 分 钟 之 后 就 想 出 了 
一 个 需要 重复 吉 历 一 个 结 反 多 次 的 算法 。 在 我 提示 出 还 有 更 快 的 算法 之 
后 ， 他 再 次 把 题目 转换 成 求 链表 的 共同 结 点 的 问题 。 期 间 在 他 解释 其 思 
路 的 过 程 中 ， 可 以 看 出 他 对 树 的 过 历 算法 理解 得 很 透彻 ， 接 下 来 写 出 的 
代码 也 很 规范 。 综 合 这 名 应 聘 者 在 本 轮 面 试 中 的 表现 ， 我 强烈 建议 我 们 
公司 录用 他 。 

如 果 面 试 官 在 面试 的 过 程 中 逐步 加 大 面试 题 的 难度 ， 通 常 对 应 聘 者 
来 说 是 件 好 事 ， 这 说 明 应 聘 者 一 开始 表现 得 很 好 ， 面 试 官 对 他 的 印象 很 
好 ， 并 很 有 兴趣 看 看 他 的 水 平 有 多 深 ， 于 是 一 步 一 步 加 大 题目 的 难度 。 
虽然 最 后 应 聘 者 可 能 不 能 很 好 地 解决 高 难度 的 问题 ， 但 最 终 仍 有 可 能 拿 
到 Offer。 与 此 相反 的 是 ， 有 些 应 聘 者 觉得 面试 的 时 候 很 多 问题 都 回答 出 
来 了 ， 可 最 终 被 拒 ， 觉 得 难以 理解 。 其 实 这 是 因为 一 开始 问 的 问题 他 回 
答 得 很 不 好 ， 面 试 官 己 经 出 判断 他 的 能 力 有 限 ， 心 里 己 经 默默 给 出 了 
NO 的 结论 。 但 为 了 照顾 应 聘 者 的 情 面 ， 也 会 问 几 个 简单 的 问题 。 虽 然 
这 些 人 简单 的 问题 应 聘 者 可 能 都 能 答对 ， 但 前 面 的 结果 已 经 不 会 改变 。 

在 这 轮 面 试 中 ， 由 于 该 应 聘 者 一 开始 的 表现 很 好 ， 我 才 决 定 加 大 难 
度 考 考 他 。 假 如 他 最 后 普通 树 中 结 点 没有 指向 父 结 点 的 指针 这 个 问题 没 
有 很 好 地 解决 ， 我 会 让 他 回头 去 写 普 通 树 中 结 点 有 指 疝 父 结 反 的 指针 这 
个 问题 的 代码 。 只 要 他 的 代码 写 得 完整 正确 ， 我 仍然 会 让 他 通过 我 的 这 
轮 面试 ， 尽 管 我 对 他 的 评价 可 能 没有 现在 这 么 高 。 






























































Table of Contents 





目录 
Be OL 
版 权 页 
tee Fe 
前 言 
第 1 章 试 的 流程 
1.1 面试 官 谈 面试 
i 试 的 三 种 形式 
121 了 也 
12.2 ”共享 元 程 面 记 
12.3 现场 面试 
13 面试 的 三 个 环节 
1.3.1 人行 WIA 
1. 应 聘 者 的 项 目 经 验 
2 M 由 Ep ERK ob 
3. HÆ KA k” 
1.3.2 面试 环节 
1.3.3 ”应聘 者 提问 环节 
1.4 Bpa 





B2% ”面试 需要 的 基础 知识 
2.1 面试 官 谈 基础 知识 
2.2 编程 语言 
231 C++ 





: IA TER 
经 暴 Æ y , E A 

口 (=i 

TJP 页 
































